Commit e825ca7f by Mark Wielaard Committed by Mark Wielaard

Connection.java (getJarFile): download and cache remote jar files.

	* gnu/gcj/protocol/jar/Connection.java (getJarFile): download and
	cache remote jar files.
	* gnu/gcj/runtime/VMClassLoader.java: Don't construct jar URL, only
	add File.separator to URL when it is a directory.
	* java/lang/ClassLoader.java: Add Classpath javadoc.
	(parent): final.
	(getParent): Add (disabled) security check.
	(findLibrary): New default method.
	* java/net/JarURLConnection.java (getManifest): Implement.
	(getInputStream): Only create InputStream when entry exists.
	(getHeaders): Only use jarFileURLConnection or JarEntry to set length
	when they exist.
	* java/net/URLClassLoader.java: New/Rewritten version from Classpath.

From-SVN: r59949
parent 24632117
2002-12-08 Mark Wielaard <mark@klomp.org>
* gnu/gcj/protocol/jar/Connection.java (getJarFile): download and
cache remote jar files.
* gnu/gcj/runtime/VMClassLoader.java: Don't construct jar URL, only
add File.separator to URL when it is a directory.
* java/lang/ClassLoader.java: Add Classpath javadoc.
(parent): final.
(getParent): Add (disabled) security check.
(findLibrary): New default method.
* java/net/JarURLConnection.java (getManifest): Implement.
(getInputStream): Only create InputStream when entry exists.
(getHeaders): Only use jarFileURLConnection or JarEntry to set length
when they exist.
* java/net/URLClassLoader.java: New/Rewritten version from Classpath.
2002-12-08 Mark Wielaard <mark@klomp.org>
* java/util/ResourceBundle.java (resourceBundleCache): Not final.
(lastDefaultLocale): New field.
(getBundle): When Locale.getDefault != lastDefaultLocale reset
......
/* Copyright (C) 1999 Free Software Foundation
/* Copyright (C) 1999, 2002 Free Software Foundation
This file is part of libgcj.
......@@ -9,19 +9,21 @@ details. */
package gnu.gcj.protocol.jar;
import java.net.URL;
import java.net.URLConnection;
import java.net.JarURLConnection;
import java.net.URLStreamHandler;
import java.net.MalformedURLException;
import java.net.ProtocolException;
import java.io.IOException;
import java.io.InputStream;
import java.io.File;
import java.io.FileOutputStream;
import java.util.jar.JarFile;
import java.util.zip.ZipFile;
import java.util.Hashtable;
/**
* Written using on-line Java Platform 1.2 API Specification.
* Status: Needs a way to download jar files and store them in the local file
* system. I don't know how to do that in a portable way. For now, it can only handle
* connections to a jar:file: url's.
*
* @author Kresten Krab Thorup <krab@gnu.org>
* @date Aug 10, 1999.
......@@ -70,14 +72,19 @@ public class Connection extends JarURLConnection
}
else
{
/*
FIXME: Here we need to download and cache the jar
file in the local file system! Stupid design. Why
can't we just create a JarFile from a bag of bytes?
*/
throw new java.io.IOException("cannot create jar file from " +
jarFileURL);
URLConnection urlconn = jarFileURL.openConnection();
InputStream is = urlconn.getInputStream();
byte[] buf = new byte[4*1024];
File f = File.createTempFile("cache", "jar");
FileOutputStream fos = new FileOutputStream(f);
int len = 0;
while((len = is.read(buf)) != -1)
fos.write(buf, 0, len);
fos.close();
// Always verify the Manifest, open read only and delete when done.
// XXX ZipFile.OPEN_DELETE not yet implemented.
// jf = new JarFile(f, true, ZipFile.OPEN_READ | ZipFile.OPEN_DELETE);
jarfile = new JarFile(f, true, ZipFile.OPEN_READ);
}
return jarfile;
......
/* Copyright (C) 1999, 2001 Free Software Foundation
/* Copyright (C) 1999, 2001, 2002 Free Software Foundation
This file is part of libgcj.
......@@ -33,23 +33,10 @@ public final class VMClassLoader extends java.net.URLClassLoader
String e = st.nextToken ();
try
{
if (e.endsWith(".jar") || e.endsWith (".zip"))
{
File archive = new File (e);
try {
p.addElement(new URL("jar", "", -1, "file://"
+ archive.getCanonicalPath ()
+ "!/"));
} catch (IOException ex) {
// empty
}
}
else if (e.endsWith ("/"))
p.addElement (new URL("file", "", -1, e));
else if (new File (e).isDirectory ())
p.addElement (new URL("file", "", -1, e + "/"));
if (!e.endsWith (File.separator) && new File (e).isDirectory ())
p.addElement (new URL("file", "", -1, e + File.separator));
else
/* Ignore path element. */;
p.addElement (new URL("file", "", -1, e));
}
catch (java.net.MalformedURLException x)
{
......
......@@ -13,7 +13,6 @@ package java.lang;
import java.io.InputStream;
import java.io.IOException;
import java.net.URL;
import java.net.URLConnection;
import java.security.AllPermission;
import java.security.CodeSource;
import java.security.Permission;
......@@ -23,13 +22,68 @@ import java.security.ProtectionDomain;
import java.util.*;
/**
* The class <code>ClassLoader</code> is intended to be subclassed by
* applications in order to describe new ways of loading classes,
* such as over the network.
* The ClassLoader is a way of customizing the way Java gets its classes
* and loads them into memory. The verifier and other standard Java things
* still run, but the ClassLoader is allowed great flexibility in determining
* where to get the classfiles and when to load and resolve them. For that
* matter, a custom ClassLoader can perform on-the-fly code generation or
* modification!
*
* <p>Every classloader has a parent classloader that is consulted before
* the 'child' classloader when classes or resources should be loaded.
* This is done to make sure that classes can be loaded from an hierarchy of
* multiple classloaders and classloaders do not accidentially redefine
* already loaded classes by classloaders higher in the hierarchy.
*
* <p>The grandparent of all classloaders is the bootstrap classloader, which
* loads all the standard system classes as implemented by GNU Classpath. The
* other special classloader is the system classloader (also called
* application classloader) that loads all classes from the CLASSPATH
* (<code>java.class.path</code> system property). The system classloader
* is responsible for finding the application classes from the classpath,
* and delegates all requests for the standard library classes to its parent
* the bootstrap classloader. Most programs will load all their classes
* through the system classloaders.
*
* <p>The bootstrap classloader in GNU Classpath is implemented as a couple of
* static (native) methods on the package private class
* <code>java.lang.VMClassLoader</code>, the system classloader is an
* instance of <code>gnu.java.lang.SystemClassLoader</code>
* (which is a subclass of <code>java.net.URLClassLoader</code>).
*
* <p>Users of a <code>ClassLoader</code> will normally just use the methods
* <ul>
* <li> <code>loadClass()</code> to load a class.</li>
* <li> <code>getResource()</code> or <code>getResourceAsStream()</code>
* to access a resource.</li>
* <li> <code>getResources()</code> to get an Enumeration of URLs to all
* the resources provided by the classloader and its parents with the
* same name.</li>
* </ul>
*
* <p>Subclasses should implement the methods
* <ul>
* <li> <code>findClass()</code> which is called by <code>loadClass()</code>
* when the parent classloader cannot provide a named class.</li>
* <li> <code>findResource()</code> which is called by
* <code>getResource()</code> when the parent classloader cannot provide
* a named resource.</li>
* <li> <code>findResources()</code> which is called by
* <code>getResource()</code> to combine all the resources with the
* same name from the classloader and its parents.</li>
* <li> <code>findLibrary()</code> which is called by
* <code>Runtime.loadLibrary()</code> when a class defined by the
* classloader wants to load a native library.</li>
* </ul>
*
* @author John Keiser
* @author Mark Wielaard
* @author Eric Blake
* @author Kresten Krab Thorup
* @see Class
* @since 1.0
* @status still missing 1.4 functionality
*/
public abstract class ClassLoader
{
/**
......@@ -73,12 +127,40 @@ public abstract class ClassLoader
// Package visible for use by Class.
Map classAssertionStatus;
private ClassLoader parent;
/**
* The classloader that is consulted before this classloader.
* If null then the parent is the bootstrap classloader.
*/
private final ClassLoader parent;
/**
* All packages defined by this classloader. It is not private in order to
* allow native code (and trusted subclasses) access to this field.
*/
private HashMap definedPackages = new HashMap();
/**
* Returns the parent of this classloader. If the parent of this
* classloader is the bootstrap classloader then this method returns
* <code>null</code>. A security check may be performed on
* <code>RuntimePermission("getClassLoader")</code>.
*
* @throws SecurityException if the security check fails
* @since 1.2
*/
public final ClassLoader getParent ()
{
/* FIXME: security */
// Check if we may return the parent classloader
SecurityManager sm = System.getSecurityManager();
if (sm != null)
{
/* FIXME: security, getClassContext() not implemented.
Class c = VMSecurityManager.getClassContext()[1];
ClassLoader cl = c.getClassLoader();
if (cl != null && cl != this)
sm.checkPermission(new RuntimePermission("getClassLoader"));
*/
}
return parent;
}
......@@ -449,7 +531,8 @@ public abstract class ClassLoader
else
{
InternalError e
= new InternalError ("unexpected exception during linking");
= new InternalError ("unexpected exception during linking: "
+ clazz.getName());
e.initCause (x);
throw e;
}
......@@ -521,6 +604,8 @@ public abstract class ClassLoader
* null when the package is not defined by this classloader or one of its
* parents.
*
* @param name the package name to find
* @return the package, if defined
* @since 1.2
*/
protected Package getPackage(String name)
......@@ -546,6 +631,7 @@ public abstract class ClassLoader
/**
* Returns all Package objects defined by this classloader and its parents.
*
* @return an array of all defined packages
* @since 1.2
*/
protected Package[] getPackages()
......@@ -578,6 +664,26 @@ public abstract class ClassLoader
}
/**
* Called by <code>Runtime.loadLibrary()</code> to get an absolute path
* to a (system specific) library that was requested by a class loaded
* by this classloader. The default implementation returns
* <code>null</code>. It should be implemented by subclasses when they
* have a way to find the absolute path to a library. If this method
* returns null the library is searched for in the default locations
* (the directories listed in the <code>java.library.path</code> system
* property).
*
* @param name the (system specific) name of the requested library
* @return the full pathname to the requested library, or null
* @see Runtime#loadLibrary()
* @since 1.2
*/
protected String findLibrary(String name)
{
return null;
}
/**
* Returns a class found in a system-specific way, typically
* via the <code>java.class.path</code> system property. Loads the
* class if necessary.
......
......@@ -134,7 +134,11 @@ public abstract class JarURLConnection extends URLConnection
if (jarfile != null)
{
// this is the easy way...
return jarfile.getInputStream (jarfile.getEntry (element));
ZipEntry entry = jarfile.getEntry(element);
if (entry != null)
return jarfile.getInputStream (entry);
else
return null;
}
else
{
......@@ -320,12 +324,17 @@ public abstract class JarURLConnection extends URLConnection
// to add others later and for consistency, we'll implement it this way.
// Add the only header we know about right now: Content-length.
long len;
long len = -1;
if (element == null)
if (jarFileURLConnection != null)
len = jarFileURLConnection.getContentLength ();
else
len = getJarEntry ().getSize ();
{
JarEntry entry = getJarEntry();
if (entry != null)
len = entry.getSize ();
}
String line = "Content-length: " + len;
hdrVec.addElement(line);
......@@ -381,7 +390,6 @@ public abstract class JarURLConnection extends URLConnection
{
JarFile file = getJarFile ();
// FIXME: implement this
return null;
return (file != null) ? file.getManifest() : null;
}
}
/* Copyright (C) 1999, 2000, 2002 Free Software Foundation
/* URLClassLoader.java -- ClassLoader that loads classes from one or more URLs
Copyright (C) 1999, 2000, 2001, 2002 Free Software Foundation, Inc.
This file is part of libgcj.
This file is part of GNU Classpath.
This software is copyrighted work licensed under the terms of the
Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
details. */
GNU Classpath is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2, or (at your option)
any later version.
GNU Classpath is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with GNU Classpath; see the file COPYING. If not, write to the
Free Software Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA
02111-1307 USA.
Linking this library statically or dynamically with other modules is
making a combined work based on this library. Thus, the terms and
conditions of the GNU General Public License cover the whole
combination.
As a special exception, the copyright holders of this library give you
permission to link this library with independent modules to produce an
executable, regardless of the license terms of these independent
modules, and to copy and distribute the resulting executable under
terms of your choice, provided that you also meet, for each linked
independent module, the terms and conditions of the license of that
module. An independent module is a module which is not derived from
or based on this library. If you modify this library, you may extend
this exception to your version of the library, but you are not
obligated to do so. If you do not wish to do so, delete this
exception statement from your version. */
package java.net;
import java.io.*;
import java.util.jar.*;
import java.util.Enumeration;
import java.util.Vector;
import java.io.ByteArrayOutputStream;
import java.io.EOFException;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FilterInputStream;
import java.io.FilePermission;
import java.io.InputStream;
import java.io.IOException;
import java.security.AccessController;
import java.security.AccessControlContext;
import java.security.CodeSource;
import java.security.SecureClassLoader;
import java.security.PrivilegedAction;
import java.security.PermissionCollection;
import java.security.cert.Certificate;
import java.util.Enumeration;
import java.util.Vector;
import java.util.HashMap;
import java.util.jar.Attributes;
import java.util.jar.JarEntry;
import java.util.jar.JarFile;
import java.util.jar.Manifest;
import java.util.zip.ZipException;
/**
* A secure class loader that can load classes and resources from
* multiple locations. Given an array of <code>URL</code>s this class
* loader will retrieve classes and resources by fetching them from
* possible remote locations. Each <code>URL</code> is searched in
* order in which it was added. If the file portion of the
* <code>URL</code> ends with a '/' character then it is interpreted
* as a base directory, otherwise it is interpreted as a jar file from
* which the classes/resources are resolved.
*
* <p>New instances can be created by two static
* <code>newInstance()</code> methods or by three public
* contructors. Both ways give the option to supply an initial array
* of <code>URL</code>s and (optionally) a parent classloader (that is
* different from the standard system class loader).</p>
*
* <p>Normally creating a <code>URLClassLoader</code> throws a
* <code>SecurityException</code> if a <code>SecurityManager</code> is
* installed and the <code>checkCreateClassLoader()</code> method does
* not return true. But the <code>newInstance()</code> methods may be
* used by any code as long as it has permission to acces the given
* <code>URL</code>s. <code>URLClassLoaders</code> created by the
* <code>newInstance()</code> methods also explicitly call the
* <code>checkPackageAccess()</code> method of
* <code>SecurityManager</code> if one is installed before trying to
* load a class. Note that only subclasses of
* <code>URLClassLoader</code> can add new URLs after the
* URLClassLoader had been created. But it is always possible to get
* an array of all URLs that the class loader uses to resolve classes
* and resources by way of the <code>getURLs()</code> method.</p>
*
* <p>Open issues:
* <ul>
*
* <li>Should the URLClassLoader actually add the locations found in
* the manifest or is this the responsibility of some other
* loader/(sub)class? (see <a
* href="http://java.sun.com/products/jdk/1.4/docs/guide/extensions/spec.html">
* Extension Mechanism Architecture - Bundles Extensions</a>)</li>
*
* <li>How does <code>definePackage()</code> and sealing work
* precisely?</li>
*
* <li>We save and use the security context (when a created by
* <code>newInstance()</code> but do we have to use it in more
* places?</li>
*
* <li>The use of <code>URLStreamHandler</code>s has not been tested.</li>
*
* </ul>
* </p>
*
* @since 1.2
*
* @author Mark Wielaard (mark@klomp.org)
* @author Wu Gansha (gansha.wu@intel.com)
*/
public class URLClassLoader extends SecureClassLoader
{
// The URLStreamHandlerFactory
URLStreamHandlerFactory factory = null;
// Class Variables
/**
* A global cache to store mappings between URLLoader and URL,
* so we can avoid do all the homework each time the same URL
* comes.
* XXX - Keeps these loaders forever which prevents garbage collection.
*/
private static HashMap urlloaders = new HashMap();
/**
* A cache to store mappings between handler factory and its
* private protocol handler cache (also a HashMap), so we can avoid
* create handlers each time the same protocol comes.
*/
private static HashMap factoryCache = new HashMap(5);
// Instance variables
/** Locations to load classes from */
private final Vector urls = new Vector();
// `path' contains simply the URL's we're using for the searching.
private Vector path;
/**
* Store pre-parsed information for each url into this vector
* each element is a URL loader, corresponding to the URL of
* the same index in "urls"
*/
private final Vector urlinfos = new Vector();
/** Factory used to get the protocol handlers of the URLs */
private final URLStreamHandlerFactory factory;
// If path[n] is a zip/jar, then this holds a JarURLConnection for
// that thing, otherwise, path[n] is null.
private Vector info;
/**
* The security context when created from <code>newInstance()</code>
* or null when created through a normal constructor or when no
* <code>SecurityManager</code> was installed.
*/
private final AccessControlContext securityContext;
private URLStreamHandler getHandler0 (String protocol)
// Helper classes
/**
* A <code>URLLoader</code> contains all logic to load resources from a
* given base <code>URL</code>.
*/
static abstract class URLLoader
{
if (factory != null)
return factory.createURLStreamHandler(protocol);
else
return null;
}
/**
* Our classloader to get info from if needed.
*/
final URLClassLoader classloader;
/**
* Createa a new URL class loader object
*
* @exception SecurityException If a security manager exists and its
* checkCreateClassLoader method doesn't allow creation of a class loader
* The base URL from which all resources are loaded.
*/
public URLClassLoader (URL[] urls)
final URL baseURL;
/**
* A <code>CodeSource</code> without any associated certificates.
* It is common for classes to not have certificates associated
* with them. If they come from the same <code>URLLoader</code>
* then it is safe to share the associated <code>CodeSource</code>
* between them since <code>CodeSource</code> is immutable.
*/
final CodeSource noCertCodeSource;
URLLoader(URLClassLoader classloader, URL baseURL)
{
this (urls, null, null);
this.classloader = classloader;
this.baseURL = baseURL;
this.noCertCodeSource = new CodeSource(baseURL, null);
}
/**
* Createa a new URL class loader object
*
* @exception SecurityException If a security manager exists and its
* checkCreateClassLoader method doesn't allow creation of a class loader
* Returns a <code>Resource</code> loaded by this
* <code>URLLoader</code>, or <code>null</code> when no
* <code>Resource</code> with the given name exists.
*/
public URLClassLoader (URL[] urls, ClassLoader parent)
abstract Resource getResource(String s);
/**
* Returns the <code>Manifest</code> associated with the
* <code>Resource</code>s loaded by this <code>URLLoader</code> or
* <code>null</code> there is no such <code>Manifest</code>.
*/
Manifest getManifest()
{
this (urls, parent, null);
return null;
}
}
// A File URL may actually be a Jar URL. Convert if possible.
private URL jarFileize (URL url)
{
if (! url.getProtocol ().equals ("jar"))
/**
* A <code>Resource</code> represents a resource in some
* <code>URLLoader</code>. It also contains all information (e.g.,
* <code>URL</code>, <code>CodeSource</code>, <code>Manifest</code> and
* <code>InputStream</code>) that is necessary for loading resources
* and creating classes from a <code>URL</code>.
*/
static abstract class Resource
{
String f = url.getFile ();
final URLLoader loader;
final String name;
// 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) != '/')
Resource(URLLoader loader, String name)
{
try
this.loader = loader;
this.name = name;
}
/**
* Returns the non-null <code>CodeSource</code> associated with
* this resource.
*/
CodeSource getCodeSource()
{
url = new URL ("jar", "", -1, (url.toExternalForm ())+"!/",
getHandler0 ("jar"));
Certificate[] certs = getCertificates();
if (certs != null)
return loader.noCertCodeSource;
else
return new CodeSource(loader.baseURL, certs);
}
catch (MalformedURLException x)
/**
* Returns <code>Certificates</code> associated with this
* resource, or null when there are none.
*/
Certificate[] getCertificates()
{
/* ignore */
return null;
}
/**
* Return a <code>URL</code> that can be used to access this resource.
*/
abstract URL getURL();
/**
* Returns the size of this <code>Resource</code> in bytes or
* <code>-1</code> when unknown.
*/
abstract int getLength();
/**
* Returns the non-null <code>InputStream</code> through which
* this resource can be loaded.
*/
abstract InputStream getInputStream() throws IOException;
}
/**
* A <code>JarURLLoader</code> is a type of <code>URLLoader</code>
* only loading from jar url.
*/
final static class JarURLLoader extends URLLoader
{
final JarFile jarfile; // The canonical jar file for this url
final URL baseJarURL; // Base jar: url for all resources loaded from jar
public JarURLLoader(URLClassLoader classloader, URL baseURL)
{
super(classloader, baseURL);
// cache url prefix for all resources in this jar url
String external = baseURL.toExternalForm();
StringBuffer sb = new StringBuffer(external.length() + 6);
sb.append("jar:");
sb.append(external);
sb.append("!/");
String jarURL = sb.toString();
URL baseJarURL = null;
JarFile jarfile = null;
try
{
baseJarURL
= new URL(null, jarURL, classloader.getURLStreamHandler("jar"));
jarfile
= ((JarURLConnection) baseJarURL.openConnection()).getJarFile();
}
return url;
catch (IOException ioe) { /* ignored */ }
this.baseJarURL = baseJarURL;
this.jarfile = jarfile;
}
protected void addURL (URL url)
/** get resource with the name "name" in the jar url */
Resource getResource(String name)
{
JarURLConnection conn = null;
// Convert a Jar File URL into Jar URL if possible.
url = jarFileize (url);
if (jarfile == null)
return null;
path.addElement (url);
JarEntry je = jarfile.getJarEntry(name);
if(je != null)
return new JarURLResource(this, name, je);
else
return null;
}
if (url.getProtocol ().equals ("jar"))
Manifest getManifest()
{
try
{
conn = (JarURLConnection) url.openConnection ();
return (jarfile == null) ? null : jarfile.getManifest();
}
catch (java.io.IOException x)
catch (IOException ioe)
{
/* ignore */
return null;
}
}
info.addElement (conn);
}
/**
* Createa a new URL class loader object
*
* @exception SecurityException If a security manager exists and its
* checkCreateClassLoader method doesn't allow creation of a class loader
*/
public URLClassLoader (URL[] urls, ClassLoader parent,
URLStreamHandlerFactory fac)
final static class JarURLResource extends Resource
{
super (parent);
private final JarEntry entry;
SecurityManager s = System.getSecurityManager();
if (s != null)
s.checkCreateClassLoader();
factory = fac;
if (urls == null || urls.length == 0)
JarURLResource(JarURLLoader loader, String name, JarEntry entry)
{
path = new Vector (1);
info = new Vector (1);
return;
super(loader, name);
this.entry = entry;
}
path = new Vector (urls.length);
info = new Vector (urls.length);
InputStream getInputStream() throws IOException
{
return ((JarURLLoader)loader).jarfile.getInputStream(entry);
}
for (int i = 0; i < urls.length; i++)
int getLength()
{
// Convert a Jar File URL into a Jar URL if possible.
URL u = jarFileize(urls[i]);
return (int)entry.getSize();
}
path.addElement (u);
Certificate[] getCertificates()
{
return entry.getCertificates();
}
if (u.getProtocol ().equals ("jar"))
URL getURL()
{
JarURLConnection conn = null;
try
{
conn = (JarURLConnection) u.openConnection ();
return new URL(((JarURLLoader)loader).baseJarURL, name,
loader.classloader.getURLStreamHandler("jar"));
}
catch (java.io.IOException x)
catch(MalformedURLException e)
{
/* ignore */
}
info.addElement (conn);
}
else
{
info.addElement (null);
throw new InternalError(e.toString());
}
}
}
public URL[] getURLs ()
/**
* Loader for remote directories.
*/
final static class RemoteURLLoader extends URLLoader
{
URL[] urls = new URL[path.size()];
path.copyInto (urls);
return urls;
final private String protocol;
RemoteURLLoader(URLClassLoader classloader, URL url)
{
super(classloader, url);
protocol = url.getProtocol();
}
/**
* Returns an Enumeration of URLs representing all of the resources on the
* URL search path having the specified name
*
* @exception IOException If an error occurs
* Get a remote resource.
* Returns null if no such resource exists.
*/
public Enumeration findResources (String name)
Resource getResource(String name)
{
Vector results = new Vector ();
for (int i = 0; i < path.size(); i++)
try
{
URL u = (URL)path.elementAt (i);
URL url = new URL(baseURL, name,
classloader.getURLStreamHandler(protocol));
URLConnection connection = url.openConnection();
try {
JarURLConnection conn = (JarURLConnection) info.elementAt (i);
// Open the connection and check the stream
// just to be sure it exists.
int length = connection.getContentLength();
InputStream stream = connection.getInputStream();
if (conn != null)
// We can do some extra checking if it is a http request
if (connection instanceof HttpURLConnection)
{
if (conn.getJarFile().getJarEntry (name) != null)
results.addElement (new URL(u, name,
getHandler0 (u.getProtocol())));
int response
= ((HttpURLConnection)connection).getResponseCode();
if (response/100 != 2)
return null;
}
if (stream != null)
return new RemoteResource(this, name, url, stream, length);
else
return null;
}
catch (IOException ioe)
{
URL p = new URL (u, name, getHandler0 (u.getProtocol()));
return null;
}
}
}
InputStream is = p.openStream();
if (is != null)
/**
* A resource from some remote location.
*/
final static class RemoteResource extends Resource
{
final private URL url;
final private InputStream stream;
final private int length;
RemoteResource(RemoteURLLoader loader, String name, URL url,
InputStream stream, int length)
{
is.close();
results.addElement (p);
super(loader, name);
this.url = url;
this.stream = stream;
this.length = length;
}
InputStream getInputStream() throws IOException
{
return stream;
}
// if we get an exception ... try the next path element
} catch (IOException x) {
continue;
public int getLength()
{
return length;
}
public URL getURL()
{
return url;
}
}
return results.elements ();
/**
* A <code>FileURLLoader</code> is a type of <code>URLLoader</code>
* only loading from file url.
*/
final static class FileURLLoader extends URLLoader
{
File dir; //the canonical file for this file url
FileURLLoader(URLClassLoader classloader, URL url)
{
super(classloader, url);
// Note that this must be a "file" protocol URL.
dir = new File(url.getFile());
}
public URL findResource (String name)
/** get resource with the name "name" in the file url */
Resource getResource(String name)
{
for (int i = 0; i < path.size(); i++)
File file = new File(dir, name);
if (file.exists() && !file.isDirectory())
return new FileResource(this, name, file);
else
return null;
}
}
final static class FileResource extends Resource
{
URL u = (URL)path.elementAt (i);
final File file;
try {
JarURLConnection conn = (JarURLConnection) info.elementAt (i);
FileResource(FileURLLoader loader, String name, File file)
{
super(loader, name);
this.file = file;
}
if (conn != null)
InputStream getInputStream() throws IOException
{
if (conn.getJarFile().getJarEntry (name) != null)
return new URL(u, name, getHandler0 (u.getProtocol()));
return new FileInputStream(file);
}
else
public int getLength()
{
URL p = new URL (u, name, getHandler0 (u.getProtocol()));
return (int)file.length();
}
InputStream is = p.openStream();
if (is != null)
public URL getURL()
{
is.close();
return p;
try
{
return new URL(loader.baseURL, name,
loader.classloader.getURLStreamHandler("file"));
}
catch(MalformedURLException e)
{
throw new InternalError(e.toString());
}
// if we get an exception ... try the next path element
} catch (IOException x) {
continue;
}
}
return null;
}
// Constructors
/**
* Finds and loads the class with the specified name from the
* URL search path
* Creates a URLClassLoader that gets classes from the supplied URLs.
* To determine if this classloader may be created the constructor of
* the super class (<code>SecureClassLoader</code>) is called first, which
* can throw a SecurityException. Then the supplied URLs are added
* in the order given to the URLClassLoader which uses these URLs to
* load classes and resources (after using the default parent ClassLoader).
*
* @exception ClassNotFoundException If the class could not be found
* @exception SecurityException if the SecurityManager disallows the
* creation of a ClassLoader.
* @param urls Locations that should be searched by this ClassLoader when
* resolving Classes or Resources.
* @see SecureClassLoader
*/
protected Class findClass (String name)
throws ClassNotFoundException
public URLClassLoader(URL[] urls) throws SecurityException
{
if (name == null)
throw new ClassNotFoundException ("null");
super();
this.factory = null;
this.securityContext = null;
addURLs(urls);
}
try
/**
* Private constructor used by the static
* <code>newInstance(URL[])</code> method. Creates an
* <code>URLClassLoader</code> without any <code>URL</code>s
* yet. This is used to bypass the normal security check for
* creating classloaders, but remembers the security context which
* will be used when defining classes. The <code>URL</code>s to
* load from must be added by the <code>newInstance()</code> method
* in the security context of the caller.
*
* @param securityContext the security context of the unprivileged code.
*/
private URLClassLoader(AccessControlContext securityContext)
{
URL url = getResource (name.replace ('.', '/') + ".class");
if (url == null)
throw new ClassNotFoundException (name);
URLConnection connection = url.openConnection ();
InputStream is = connection.getInputStream ();
super();
this.factory = null;
this.securityContext = securityContext;
}
int len = connection.getContentLength ();
byte[] data = new byte[len];
/**
* Creates a <code>URLClassLoader</code> that gets classes from the supplied
* <code>URL</code>s.
* To determine if this classloader may be created the constructor of
* the super class (<code>SecureClassLoader</code>) is called first, which
* can throw a SecurityException. Then the supplied URLs are added
* in the order given to the URLClassLoader which uses these URLs to
* load classes and resources (after using the supplied parent ClassLoader).
* @exception SecurityException if the SecurityManager disallows the
* creation of a ClassLoader.
* @exception SecurityException
* @param urls Locations that should be searched by this ClassLoader when
* resolving Classes or Resources.
* @param parent The parent class loader used before trying this class
* loader.
* @see SecureClassLoader
*/
public URLClassLoader(URL[] urls, ClassLoader parent)
throws SecurityException
{
super(parent);
this.factory = null;
this.securityContext = null;
addURLs(urls);
}
int left = len;
int off = 0;
while (left > 0)
/**
* Private constructor used by the static
* <code>newInstance(URL[])</code> method. Creates an
* <code>URLClassLoader</code> with the given parent but without any
* <code>URL</code>s yet. This is used to bypass the normal security
* check for creating classloaders, but remembers the security
* context which will be used when defining classes. The
* <code>URL</code>s to load from must be added by the
* <code>newInstance()</code> method in the security context of the
* caller.
*
* @param securityContext the security context of the unprivileged code.
*/
private URLClassLoader(ClassLoader parent,
AccessControlContext securityContext)
{
int c = is.read (data, off, len-off);
if (c == -1 || c == 0)
throw new InternalError ("premature end of file");
left -= c;
off += c;
super(parent);
this.factory = null;
this.securityContext = securityContext;
}
// Now construct the CodeSource (if loaded from a jar file)
CodeSource source = null;
if (url.getProtocol().equals("jar"))
/**
* Creates a URLClassLoader that gets classes from the supplied URLs.
* To determine if this classloader may be created the constructor of
* the super class (<CODE>SecureClassLoader</CODE>) is called first, which
* can throw a SecurityException. Then the supplied URLs are added
* in the order given to the URLClassLoader which uses these URLs to
* load classes and resources (after using the supplied parent ClassLoader).
* It will use the supplied <CODE>URLStreamHandlerFactory</CODE> to get the
* protocol handlers of the supplied URLs.
* @exception SecurityException if the SecurityManager disallows the
* creation of a ClassLoader.
* @exception SecurityException
* @param urls Locations that should be searched by this ClassLoader when
* resolving Classes or Resources.
* @param parent The parent class loader used before trying this class
* loader.
* @param factory Used to get the protocol handler for the URLs.
* @see SecureClassLoader
*/
public URLClassLoader(URL[] urls,
ClassLoader parent,
URLStreamHandlerFactory factory)
throws SecurityException
{
Certificate[] certificates =
((JarURLConnection) connection).getCertificates();
String u = url.toExternalForm ();
u = u.substring (4); //skip "jar:"
int i = u.indexOf ('!');
if (i >= 0)
u = u.substring (0, i);
url = new URL(u);
super(parent);
this.securityContext = null;
this.factory = factory;
addURLs(urls);
source = new CodeSource(url, certificates);
// If this factory is still not in factoryCache, add it,
// since we only support three protocols so far, 5 is enough
// for cache initial size
synchronized(factoryCache)
{
if(factory != null && factoryCache.get(factory) == null)
factoryCache.put(factory, new HashMap(5));
}
}
else if (url.getProtocol().equals("file"))
// Methods
/**
* Adds a new location to the end of the internal URL store.
* @param newUrl the location to add
*/
protected void addURL(URL newUrl)
{
try
synchronized(urlloaders)
{
String u = url.toExternalForm();
// Skip "file:" and then get canonical directory name.
File f = new File(u.substring(5));
f = f.getCanonicalFile();
url = new URL("file", "", f.getParent());
source = new CodeSource (url, null);
}
catch (IOException ignore)
if (newUrl == null)
return; // Silently ignore...
// check global cache to see if there're already url loader
// for this url
URLLoader loader = (URLLoader)urlloaders.get(newUrl);
if (loader == null)
{
}
String file = newUrl.getFile();
// Check that it is not a directory
if (!(file.endsWith("/") || file.endsWith(File.separator)))
loader = new JarURLLoader(this, newUrl);
else // it's a url that point to a jar file
if ("file".equals(newUrl.getProtocol()))
loader = new FileURLLoader(this, newUrl);
else
loader = new RemoteURLLoader(this, newUrl);
// cache it
urlloaders.put(newUrl, loader);
}
return defineClass (name, data, 0, len, source);
urls.add(newUrl);
urlinfos.add(loader);
}
}
catch (java.io.IOException x)
/**
* Adds an array of new locations to the end of the internal URL store.
* @param newUrls the locations to add
*/
private void addURLs(URL[] newUrls)
{
throw new ClassNotFoundException(name);
for (int i = 0; i < newUrls.length; i++)
{
addURL(newUrls[i]);
}
}
......@@ -362,13 +731,12 @@ public class URLClassLoader extends SecureClassLoader
// Look if the Manifest indicates that this package is sealed
// XXX - most likely not completely correct!
// Shouldn't we also check the sealed attribute of the complete jar?
// http://java.sun.com/products/jdk/1.3/docs/guide/extensions/spec.html#bundled
// http://java.sun.com/products/jdk/1.4/docs/guide/extensions/spec.html#bundled
// But how do we get that jar manifest here?
String sealed = attr.getValue(Attributes.Name.SEALED);
if ("false".equals(sealed))
{
// Make sure that the URL is null so the package is not
// sealed.
// make sure that the URL is null so the package is not sealed
url = null;
}
......@@ -377,21 +745,217 @@ public class URLClassLoader extends SecureClassLoader
}
/**
* Returns the permissions needed to access a particular code source.
* These permissions includes those returned by
* <CODE>SecureClassLoader.getPermissions</CODE> and the actual permissions
* to access the objects referenced by the URL of the code source.
* The extra permissions added depend on the protocol and file portion of
* the URL in the code source. If the URL has the "file" protocol ends with
* a / character then it must be a directory and a file Permission to read
* everthing in that directory and all subdirectories is added. If the URL
* had the "file" protocol and doesn't end with a / character then it must
* be a normal file and a file permission to read that file is added. If the
* URL has any other protocol then a socket permission to connect and accept
* connections from the host portion of the URL is added.
* Finds (the first) class by name from one of the locations. The locations
* are searched in the order they were added to the URLClassLoader.
*
* @param className the classname to find
* @exception ClassNotFoundException when the class could not be found or
* loaded
* @return a Class object representing the found class
*/
protected Class findClass(final String className)
throws ClassNotFoundException
{
// Just try to find the resource by the (almost) same name
String resourceName = className.replace('.', '/') + ".class";
Resource resource = findURLResource(resourceName);
if (resource == null)
throw new ClassNotFoundException(className + " not found in " + urls);
// Try to read the class data, create the CodeSource, Package and
// construct the class (and watch out for those nasty IOExceptions)
try
{
byte [] data;
InputStream in = resource.getInputStream();
int length = resource.getLength();
if (length != -1)
{
// We know the length of the data.
// Just try to read it in all at once
data = new byte[length];
int pos = 0;
while(length - pos > 0)
{
int len = in.read(data, pos, length - pos);
if (len == -1)
throw new EOFException("Not enough data reading from: "
+ in);
pos += len;
}
}
else
{
// We don't know the data length.
// Have to read it in chunks.
ByteArrayOutputStream out = new ByteArrayOutputStream(4096);
byte b[] = new byte[4096];
int l = 0;
while (l != -1)
{
l = in.read(b);
if (l != -1)
out.write(b, 0, l);
}
data = out.toByteArray();
}
final byte[] classData = data;
// Now get the CodeSource
final CodeSource source = resource.getCodeSource();
// Find out package name
String packageName = null;
int lastDot = className.lastIndexOf('.');
if (lastDot != -1)
packageName = className.substring(0, lastDot);
if (packageName != null && getPackage(packageName) == null)
{
// define the package
Manifest manifest = resource.loader.getManifest();
if (manifest == null)
definePackage(packageName,
null, null, null, null, null, null, null);
else
definePackage(packageName, manifest, resource.loader.baseURL);
}
// And finally construct the class!
SecurityManager sm = System.getSecurityManager();
if (sm != null && securityContext != null)
{
return (Class)AccessController.doPrivileged
(new PrivilegedAction()
{
public Object run()
{
return defineClass(className, classData,
0, classData.length,
source);
}
}, securityContext);
}
else
return defineClass(className, classData,
0, classData.length,
source);
}
catch (IOException ioe)
{
throw new ClassNotFoundException(className, ioe);
}
}
/**
* Finds the first occurrence of a resource that can be found. The locations
* are searched in the order they were added to the URLClassLoader.
*
* @param resourceName the resource name to look for
* @return the URLResource for the resource if found, null otherwise
*/
private Resource findURLResource(String resourceName)
{
int max = urls.size();
for (int i = 0; i < max; i++)
{
URLLoader loader = (URLLoader)urlinfos.elementAt(i);
if (loader == null)
continue;
Resource resource = loader.getResource(resourceName);
if (resource != null)
return resource;
}
return null;
}
/**
* Finds the first occurrence of a resource that can be found.
*
* @param resourceName the resource name to look for
* @return the URL if found, null otherwise
*/
public URL findResource(String resourceName)
{
Resource resource = findURLResource(resourceName);
if (resource != null)
return resource.getURL();
// Resource not found
return null;
}
/**
* If the URLStreamHandlerFactory has been set this return the appropriate
* URLStreamHandler for the given protocol, if not set returns null.
*
* @param protocol the protocol for which we need a URLStreamHandler
* @return the appropriate URLStreamHandler or null
*/
URLStreamHandler getURLStreamHandler(String protocol)
{
if (factory == null)
return null;
URLStreamHandler handler;
synchronized (factoryCache)
{
// check if there're handler for the same protocol in cache
HashMap cache = (HashMap)factoryCache.get(factory);
handler = (URLStreamHandler)cache.get(protocol);
if(handler == null)
{
// add it to cache
handler = factory.createURLStreamHandler(protocol);
cache.put(protocol, handler);
}
}
return handler;
}
/**
* Finds all the resources with a particular name from all the locations.
*
* @exception IOException when an error occurs accessing one of the
* locations
* @param resourceName the name of the resource to lookup
* @return a (possible empty) enumeration of URLs where the resource can be
* found
*/
public Enumeration findResources(String resourceName) throws IOException
{
Vector resources = new Vector();
int max = urls.size();
for (int i = 0; i < max; i++)
{
URLLoader loader = (URLLoader)urlinfos.elementAt(i);
Resource resource = loader.getResource(resourceName);
if (resource != null)
resources.add(resource.getURL());
}
return resources.elements();
}
/**
* Returns the permissions needed to access a particular code
* source. These permissions includes those returned by
* <code>SecureClassLoader.getPermissions()</code> and the actual
* permissions to access the objects referenced by the URL of the
* code source. The extra permissions added depend on the protocol
* and file portion of the URL in the code source. If the URL has
* the "file" protocol ends with a '/' character then it must be a
* directory and a file Permission to read everything in that
* directory and all subdirectories is added. If the URL had the
* "file" protocol and doesn't end with a '/' character then it must
* be a normal file and a file permission to read that file is
* added. If the <code>URL</code> has any other protocol then a
* socket permission to connect and accept connections from the host
* portion of the URL is added.
*
* @param source The codesource that needs the permissions to be accessed
* @return the collection of permissions needed to access the code resource
* @see SecureClassLoader.getPermissions()
* @see java.security.SecureClassLoader#getPermissions()
*/
protected PermissionCollection getPermissions(CodeSource source)
{
......@@ -408,7 +972,7 @@ public class URLClassLoader extends SecureClassLoader
{
String file = url.getFile();
// If the file end in / it must be an directory
if (file.endsWith("/"))
if (file.endsWith("/") || file.endsWith(File.separator))
{
// Grant permission to read everything in that directory and
// all subdirectories
......@@ -425,6 +989,7 @@ public class URLClassLoader extends SecureClassLoader
{
// Grant permission to connect to and accept connections from host
String host = url.getHost();
if (host != null)
permissions.add(new SocketPermission(host, "connect,accept"));
}
......@@ -432,27 +997,72 @@ public class URLClassLoader extends SecureClassLoader
}
/**
* Creates a new instance of a URLClassLoader that gets classes from the
* supplied URLs. This class loader will have as parent the standard
* system class loader.
* @param urls the initial URLs used to resolve classes and resources
* Returns all the locations that this class loader currently uses the
* resolve classes and resource. This includes both the initially supplied
* URLs as any URLs added later by the loader.
* @return All the currently used URLs
*/
public static URLClassLoader newInstance(URL[] urls) throws
SecurityException
public URL[] getURLs()
{
return new URLClassLoader(urls);
return (URL[]) urls.toArray(new URL[urls.size()]);
}
/**
* Creates a new instance of a URLClassLoader that gets classes from the
* supplied URLs and with the supplied loader as parent class loader.
* @param urls the initial URLs used to resolve classes and resources
* Creates a new instance of a <code>URLClassLoader</code> that gets
* classes from the supplied <code>URL</code>s. This class loader
* will have as parent the standard system class loader.
*
* @param urls the initial URLs used to resolve classes and
* resources
*
* @exception SecurityException when the calling code does not have
* permission to access the given <code>URL</code>s
*/
public static URLClassLoader newInstance(URL urls[])
throws SecurityException
{
return newInstance(urls, null);
}
/**
* Creates a new instance of a <code>URLClassLoader</code> that gets
* classes from the supplied <code>URL</code>s and with the supplied
* loader as parent class loader.
*
* @param urls the initial URLs used to resolve classes and
* resources
* @param parent the parent class loader
*
* @exception SecurityException when the calling code does not have
* permission to access the given <code>URL</code>s
*/
public static URLClassLoader newInstance(URL[] urls,
ClassLoader parent)
public static URLClassLoader newInstance(URL urls[],
final ClassLoader parent)
throws SecurityException
{
SecurityManager sm = System.getSecurityManager();
if (sm == null)
return new URLClassLoader(urls, parent);
else
{
final Object securityContext = sm.getSecurityContext();
// XXX - What to do with anything else then an AccessControlContext?
if (!(securityContext instanceof AccessControlContext))
throw new SecurityException
("securityContext must be AccessControlContext: "
+ securityContext);
URLClassLoader loader =
(URLClassLoader)AccessController.doPrivileged(new PrivilegedAction()
{
public Object run()
{
return new URLClassLoader
(parent, (AccessControlContext)securityContext);
}
});
loader.addURLs(urls);
return loader;
}
}
}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment