public class PreferredClassProvider extends RMIClassLoaderSpi
RMIClassLoader
provider that supports preferred
classes.
See the RMIClassLoader
specification for information
about how to install and configure the RMIClassLoader
service provider.
PreferredClassProvider
uses instances of PreferredClassLoader
to load classes from codebase URI paths
supplied to RMIClassLoader.loadClass
methods. In
previous releases only codebase URL paths were permitted.
PreferredClassProvider
does not enforce DownloadPermission
by default, but a subclass can configure it to
do so by passing true
as the argument to the
protected
constructor.
By overriding the getClassAnnotation(ClassLoader)
method, a subclass can also
configure the class annotations to be used for classes defined by
the system class loader, its ancestor class loaders, and any class
loader that is not an instance of ClassAnnotation
or URLClassLoader
.
PreferredClassProvider
implements the abstract
methods of RMIClassLoaderSpi
. Where applicable, these
definitions and descriptions are relative to the instance of
PreferredClassProvider
on which a method is invoked
and the context in which it is invoked.
The annotation string for a class loader is determined by the following procedure:
getClassAnnotation(ClassLoader)
with the loader.
ClassAnnotation
, the annotation string is the result of invoking
getClassAnnotation
on
the loader.
URLClassLoader
, the annotation string is a space-separated list of
the URLs returned by an invocation of getURLs
on the loader.
getClassAnnotation(ClassLoader)
with the loader.
URL(String)
constructor; if such
parsing would result in a MalformedURLException
, then the
annotation URL path for the loader is only defined to the extent
that it is not equal to any other path of URLs.
A PreferredClassProvider
maintains an internal
table of class loader instances indexed by keys that comprise a
path of URIs and a parent class loader. In previous releases keys utilised
URL
, but now utilise Uri
by default. The following property
-Dnet.jini.loader.codebaseAnnotation=URL
may be set from the command line to revert to URL
. The table does not
strongly reference the class loader instances, in order to allow
them (and the classes they have defined) to be garbage collected
when they are not otherwise reachable.
The behavioural difference between Uri
and URL
when used
in ClassLoader index keys is subtle, URL
remote links rely on DNS to resolve
domain names to IP addresses, for this reason, when using strict URL
codebase annotations, the IP address of each codebase at the time they're resolved
is part of the codebase annotations identity. Uri
identity on the other hand is
determined by RFC3986 normalization and is more flexible the codebase server
to change its IP address or be replicated by other codebase servers
different IP addresses, provided they can be reached by their domain name
address.
The methods loadClass
, loadProxyClass
, and getClassLoader
, which
each have a String
parameter named
codebase
, have the following behaviors in common:
codebase
may be null
. If it is not
null
, it is interpreted as a path of URLs by parsing
it as a list of URLs separated by spaces. It is recommended that
URLs be compliant with RFC3986 Syntax. Prior to parsing, any file path
separators converted to '/' and any illegal characters are percentage escaped,
Uri
is used to parse each URL
in compliance with RFC3986, in addition file URL paths are
converted to upper case for case insensitive file systems. The array of
RFC3986 normalised Uris along with the current threads context ClassLoader
is used to locate the correct ClassLoader. After normalisation is complete,
each URL is parsed with the URL(String)
constructor; this could
result in a MalformedURLException
. This path of URLs is the
codebase URL path for the invocation.
codebase
and the current thread's context
class loader as follows. If codebase
is
null
, then the codebase loader is the current thread's
context class loader. Otherwise, for each non-null
loader starting with the current thread's context class loader and
continuing with each successive parent class loader, if the
codebase Uri RFC3986 normalised path is equal to the loader's annotation
Uri RFC3986 normalised path, then the codebase loader is that loader.
If no such matching loader is found, then the codebase loader is the loader
in this PreferredClassProvider
's internal table with the
codebase Uri RFC3986 normalised path as the key's path of URLs and the current
thread's context class loader as the key's parent class loader. If
no such entry exists in the table, then one is created by invoking
createClassLoader
with the codebase URL
path, the current thread's context class loader, and the
boolean
requireDlPerm
value that this
PreferredClassProvider
was constructed with; the
created loader is added to the table, and it is chosen as the
codebase loader.
openConnection()
.getPermission()
on the URL
object is not a FilePermission
or if it is a FilePermission
whose
name does not contain a directory separator, then that permission
is the appropriate permission. If it is a
FilePermission
whose name contains a directory
separator, then the appropriate permission is a
FilePermission
with action "read"
and the
same name except with the last path segment replaced with
"-"
(that is, permission to read all files in the same
directory and all subdirectories).
When PreferredClassProvider
attempts to load a
class (or interface) named N
using class loader
L
, it does so in a manner equivalent to
evaluating the following expression:
Class.forName(In particular, the case ofN
, false,L
)
N
being the binary
name of an array class is supported.This implementation uses the Logger
named
net.jini.loader.pref.PreferredClassProvider
to log
information at the following levels:
Level | Description |
---|---|
FAILED | class loading failures |
HANDLED | exceptions caught during class loading operations |
FINE | invocations of loadClass and loadProxyClass
|
FINEST | detailed activity of
loadClass and loadProxyClass
implementations
|
Modifier and Type | Class and Description |
---|---|
private static class |
PreferredClassProvider.LoaderKey
Loader table key: a codebase annotation and a weak reference to
a parent class loader (possibly null).
|
Modifier and Type | Field and Description |
---|---|
private java.util.concurrent.ConcurrentMap<ClassLoader,PermissionCollection> |
classLoaderPerms
Map to hold permissions needed to check the URLs of
URLClassLoader objects.
|
private static String |
codebaseProperty
value of "java.rmi.server.codebase" property, as cached at class
initialization time.
|
private static Permission |
getClassLoaderPermission |
private boolean |
initialized
true if constructor has completed successfully
|
private static java.util.concurrent.ConcurrentMap<PreferredClassProvider.LoaderKey,ClassLoader> |
loaderTable
table mapping codebase Uri path and context class loader pairs
to class loader instances.
|
private static Set<ClassLoader> |
localLoaders
table of "local" class loaders
|
private static Logger |
logger
provider logger
|
private static String |
PRIMITIVE_TYPES
encodings for primitive array class element types
|
private boolean |
requireDlPerm
download from codebases with no dl perm allowed?
|
private static boolean |
uri
value of "net.jini.loader.codebaseAnnotation" property, as cached at class
initialization time.
|
private static java.util.concurrent.ConcurrentMap<String,Uri[]> |
uriCache |
private static java.util.concurrent.ConcurrentMap<List<Uri>,URL[]> |
urlCache
URL cache is time based, we need this to be as fast as possible,
every remote class to be loaded is annotated.
|
Modifier | Constructor and Description |
---|---|
|
PreferredClassProvider()
Creates a new
PreferredClassProvider . |
protected |
PreferredClassProvider(boolean requireDlPerm)
Creates a new
PreferredClassProvider . |
Modifier and Type | Method and Description |
---|---|
private URL[] |
asURL(Uri[] uris)
Converts Uri[] to URL[].
|
private void |
checkInitialized() |
private void |
checkLoader(ClassLoader loader,
ClassLoader parent,
Uri[] uris,
URL[] urls) |
protected ClassLoader |
createClassLoader(URL[] urls,
ClassLoader parent,
boolean requireDlPerm)
Creates the class loader for this
PreferredClassProvider to use to load classes from
the specified path of URLs with the specified delegation
parent. |
private ClassLoader |
findOriginLoader(Uri[] pathURLs,
ClassLoader parent)
Return the origin class loader for the
pathURLs or
null if the origin loader was not present in the delegation
hierarchy. |
private ClassLoader |
findOriginLoader0(Uri[] pathURLs,
ClassLoader parent) |
String |
getClassAnnotation(Class cl)
Provides the implementation for
RMIClassLoaderSpi.getClassAnnotation(Class) . |
protected String |
getClassAnnotation(ClassLoader loader)
Returns the annotation string for the specified class loader.
|
private static ClassLoader |
getClassLoader(Class c) |
ClassLoader |
getClassLoader(String codebase)
Provides the implementation for
RMIClassLoaderSpi.getClassLoader(String) . |
private String |
getLoaderAnnotation(ClassLoader loader,
boolean check)
Returns the annotation string for the specified class loader
(possibly null).
|
private Uri[] |
getLoaderAnnotationURIs(ClassLoader loader)
Returns an array of URLs corresponding to the annotation string
for the specified class loader, or null if the annotation
string is null.
|
private static String |
getProxySuccessLogMessage(SecurityManager sm,
SecurityException secEx) |
private static ClassLoader |
getRMIContextClassLoader()
Return the class loader to be used as the parent for an RMI class
loader used in the current execution context.
|
private boolean |
interfacePreferred(PreferredClassLoader codebaseLoader,
String[] interfaceNames,
String codebase)
Returns true if at least one of the specified interface names
is preferred for the specified class loader; returns false if
none of them are preferred.
|
private static boolean |
isLocalLoader(ClassLoader loader)
Return true if the given loader is the system class loader or
its parent (i.e. the loader for installed extensions) or the null
class loader
|
Class |
loadClass(String codebase,
String name,
ClassLoader defaultLoader)
Provides the implementation for
RMIClassLoaderSpi.loadClass(String,String,ClassLoader) . |
private Class |
loadProxyClass(String[] interfaceNames,
ClassLoader interfaceLoader,
String interfaceLoaderName,
ClassLoader otherLoader,
boolean tryOtherLoaderFirst)
Attempts to load the specified interfaces by name using the
specified loader, and if that is successful, attempts to get a
dynamic proxy class that implements those interfaces.
|
Class |
loadProxyClass(String codebase,
String[] interfaceNames,
ClassLoader defaultLoader)
Provides the implementation of
RMIClassLoaderSpi.loadProxyClass(String,String[],ClassLoader) . |
private ClassLoader |
loadProxyInterfaces(String[] interfaces,
ClassLoader loader,
Class[] classObjs,
boolean[] useNonpublic) |
private ClassLoader |
lookupLoader(Uri[] uris,
URL[] urls,
ClassLoader parent)
Look up the class loader for the given codebase URL path
and the given parent class loader.
|
private static Uri[] |
pathToURIs(String path)
Convert a string containing a space-separated list of URL Strings into a
corresponding array of Uri objects, throwing a MalformedURLException
if any of the URLs are invalid.
|
private boolean |
urlsMatchLoaderAnnotation(Uri[] urls,
ClassLoader loader)
Returns true if the specified path of URLs is equal to the
annotation URLs of the specified loader, and false otherwise.
|
private static final boolean uri
private static final String PRIMITIVE_TYPES
private final boolean requireDlPerm
private final boolean initialized
private static final Logger logger
private static final Permission getClassLoaderPermission
private static String codebaseProperty
private static final Set<ClassLoader> localLoaders
private static final java.util.concurrent.ConcurrentMap<PreferredClassProvider.LoaderKey,ClassLoader> loaderTable
private static final java.util.concurrent.ConcurrentMap<List<Uri>,URL[]> urlCache
private static final java.util.concurrent.ConcurrentMap<String,Uri[]> uriCache
private final java.util.concurrent.ConcurrentMap<ClassLoader,PermissionCollection> classLoaderPerms
public PreferredClassProvider()
PreferredClassProvider
.
This constructor is used by the RMIClassLoader
service provider location mechanism when
PreferredClassProvider
is configured as the
RMIClassLoader
provider class.
If there is a security manager, its checkCreateClassLoader
method is invoked; this could result in a
SecurityException
.
DownloadPermission
is not enforced by the created
provider.
SecurityException
- if there is a security manager and
the invocation of its checkCreateClassLoader
method failsprotected PreferredClassProvider(boolean requireDlPerm)
PreferredClassProvider
.
This constructor is used by subclasses to control whether
or not DownloadPermission
is enforced.
If there is a security manager, its checkCreateClassLoader
method is invoked; this could result in a
SecurityException
.
requireDlPerm
- if true
, the class loaders
created by the provider will only define classes with a CodeSource
that is granted DownloadPermission
SecurityException
- if there is a security manager and
the invocation of its checkCreateClassLoader
method failsprivate void checkInitialized()
private void checkLoader(ClassLoader loader, ClassLoader parent, Uri[] uris, URL[] urls)
public Class loadClass(String codebase, String name, ClassLoader defaultLoader) throws MalformedURLException, ClassNotFoundException
RMIClassLoaderSpi.loadClass(String,String,ClassLoader)
.
PreferredClassProvider
implements this method
as follows:
If name
is the binary name of an array class
(of one or more dimensions) with a primitive element type, this
method returns the Class
for that array class.
Otherwise, if defaultLoader
is not
null
and any of the following conditions are true:
codebase
is null
.
defaultLoader
.
PreferredClassLoader
.
PreferredClassLoader
and an invocation of isPreferredResource
on the codebase loader with the class name described below as
the first argument and true
as the second argument
returns false
. If name
is the binary
name of an array class (of one or more dimensions) with a
element type that is a reference type, the class name passed to
isPreferredResource
is the binary name of that
element type; otherwise, the class name passed to
isPreferredResource
is name
. This
invocation is only done if none of the previous conditions are
true. If isPreferredResource
throws an
IOException
, this method throws a
ClassNotFoundException
.
defaultLoader
. If this attempt
succeeds, this method returns the resulting Class
;
if it throws a ClassNotFoundException
, this method
proceeds as follows.
Otherwise, this method attempts to load the class with the
specified name using the codebase loader, if there is a
security manager and the current security context has
permission to access the codebase loader, or using the current
thread's context class loader otherwise. If this attempt
succeeds, this method returns the resulting Class
;
if it throws a ClassNotFoundException
, this method
throws a ClassNotFoundException
.
loadClass
in class RMIClassLoaderSpi
codebase
- the codebase URL path as a space-separated list
of URLs, or null
name
- the binary name of the class to loaddefaultLoader
- additional contextual class loader
to use, or null
Class
object representing the loaded classMalformedURLException
- if codebase
is
non-null
and contains an invalid URLClassNotFoundException
- if a definition for the class
could not be loadedpublic String getClassAnnotation(Class cl)
RMIClassLoaderSpi.getClassAnnotation(Class)
.
PreferredClassProvider
implements this method
as follows:
If cl
is an array class (of one or more
dimensions) with a primitive element type, this method returns
null
.
Otherwise, this method returns the annotation string for the
defining class loader of cl
, except that if the
annotation string would be determined by an invocation of
URLClassLoader.getURLs
on that
loader and the current security context does not have the
permissions necessary to connect to each URL returned by that
invocation (where the permission to connect to a URL is
determined by invoking openConnection()
.getPermission()
on the URL
object), this method
returns the result of invoking getClassAnnotation(ClassLoader)
with the loader instead.
getClassAnnotation
in class RMIClassLoaderSpi
cl
- the class to obtain the annotation string fornull
protected String getClassAnnotation(ClassLoader loader)
This method is invoked in order to determine the annotation
string for the system class loader, an ancestor of the system
class loader, any class loader that is not an instance of
ClassAnnotation
or URLClassLoader
, or (for an
invocation of getClassAnnotation(Class)
) a URLClassLoader
for
which the current security context does not have the
permissions necessary to connect to all of its URLs.
PreferredClassProvider
implements this method
as follows:
This method returns the value of the system property
"java.rmi.server.codebase"
(or possibly an earlier
cached value).
loader
- the class loader to obtain the annotation string
fornull
private String getLoaderAnnotation(ClassLoader loader, boolean check)
private static boolean isLocalLoader(ClassLoader loader)
public ClassLoader getClassLoader(String codebase) throws MalformedURLException
RMIClassLoaderSpi.getClassLoader(String)
.
PreferredClassProvider
implements this method
as follows:
If there is a security manager, its
checkPermission
method is invoked with a
RuntimePermission("getClassLoader")
permission;
this could result in a SecurityException
. Also,
if there is a security manager, the codebase loader is not the
current thread's context class loader, and the current security
context does not have permission to access the codebase loader,
this method throws a SecurityException
.
This method returns the codebase loader if there is a security manager, or the current thread's context class loader otherwise.
getClassLoader
in class RMIClassLoaderSpi
codebase
- the codebase URL path as a space-separated list
of URLs, or null
MalformedURLException
- if codebase
is
non-null
and contains an invalid URLSecurityException
- if there is a security manager and
the invocation of its checkPermission
method
fails, or if the current security context does not have the
permissions necessary to connect to all of the URLs in the
codebase URL pathpublic Class loadProxyClass(String codebase, String[] interfaceNames, ClassLoader defaultLoader) throws MalformedURLException, ClassNotFoundException
RMIClassLoaderSpi.loadProxyClass(String,String[],ClassLoader)
.
PreferredClassProvider
implements this method
as follows:
If defaultLoader
is not null
and
any of the following conditions are true:
codebase
is null
.
defaultLoader
.
PreferredClassLoader
.
PreferredClassLoader
and an invocation of isPreferredResource
on the codebase loader for each element of
interfaces
, with the element as the first argument
and true
as the second argument, all return
false
. These invocations are only done if none of
the previous conditions are true. If any invocation of
isPreferredResource
throws an
IOException
, this method throws a
ClassNotFoundException
.
interfaces
using
defaultLoader
. If all of the interfaces are
loaded successfully, then
public
: if
there is a security manager, the codebase loader is the current
thread's context class loader or the current security context
has permission to access the codebase loader, and the
annotation URL path for the codebase loader is not equal to the
annotation URL path for defaultLoader
, this method
first attempts to get a dynamic proxy class (using Proxy.getProxyClass
) that is defined by
the codebase loader and that implements all of the interfaces,
and if this attempt succeeds, this method returns the resulting
Class
. Otherwise, this method attempts to get a
dynamic proxy class that is defined by
defaultLoader
and that implements all of the
interfaces. If that attempt succeeds, this method returns the
resulting Class
; if it throws an
IllegalArgumentException
, this method throws a
ClassNotFoundException
.
public
interfaces are
defined by the same class loader: this method attempts to get a
dynamic proxy class that is defined by that loader and that
implements all of the interfaces. If this attempt succeeds,
this method returns the resulting Class
; if it
throws an IllegalArgumentException
, this method
throws a ClassNotFoundException
.
public
interfaces defined by different class loaders): this method
throws a LinkageError
.
ClassNotFoundException
, this method proceeds as
follows.
Otherwise, this method attempts to load all of the
interfaces named by the elements of interfaces
using the codebase loader, if there is a security manager and
the current security context has permission to access the
codebase loader, or using the current thread's context class
loader otherwise. If all of the interfaces are loaded
successfully, then
public
:
this method attempts to get a dynamic proxy class that is
defined by the loader used to load the interfaces and that
implements all of the interfaces. If this attempt succeeds,
this method returns the resulting Class
; if it
throws an IllegalArgumentException
, this method
throws a ClassNotFoundException
.
public
interfaces are
defined by the same class loader: this method attempts to get a
dynamic proxy class that is defined by that loader and that
implements all of the interfaces. If this attempt succeeds,
this method returns the resulting Class
; if it
throws an IllegalArgumentException
, this method
throws a ClassNotFoundException
.
public
interfaces defined by different class loaders): this method
throws a LinkageError
.
ClassNotFoundException
, this method throws a
ClassNotFoundException
.loadProxyClass
in class RMIClassLoaderSpi
codebase
- the codebase URL path as a space-separated list
of URLs, or null
interfaceNames
- the binary names of the interfaces for
the proxy class to implementdefaultLoader
- additional contextual class loader to use,
or null
MalformedURLException
- if codebase
is
non-null
and contains an invalid URLClassNotFoundException
- if a definition for one of the
named interfaces could not be loaded, or if creation of the
dynamic proxy class failed (such as if
Proxy.getProxyClass
would throw an
IllegalArgumentException
for the given interface
list)private static String getProxySuccessLogMessage(SecurityManager sm, SecurityException secEx)
private Class loadProxyClass(String[] interfaceNames, ClassLoader interfaceLoader, String interfaceLoaderName, ClassLoader otherLoader, boolean tryOtherLoaderFirst) throws ClassNotFoundException
ClassNotFoundException
private boolean interfacePreferred(PreferredClassLoader codebaseLoader, String[] interfaceNames, String codebase) throws ClassNotFoundException
ClassNotFoundException
private Uri[] getLoaderAnnotationURIs(ClassLoader loader) throws MalformedURLException
MalformedURLException
private boolean urlsMatchLoaderAnnotation(Uri[] urls, ClassLoader loader)
private ClassLoader loadProxyInterfaces(String[] interfaces, ClassLoader loader, Class[] classObjs, boolean[] useNonpublic) throws ClassNotFoundException
ClassNotFoundException
private static Uri[] pathToURIs(String path) throws MalformedURLException
path
- the string path to be converted to an array of urlsMalformedURLException
- if the string path of urls contains a
mal-formed url which can not be converted into a url object.private URL[] asURL(Uri[] uris) throws MalformedURLException
MalformedURLException
private static ClassLoader getRMIContextClassLoader()
private ClassLoader findOriginLoader(Uri[] pathURLs, ClassLoader parent)
pathURLs
or
null if the origin loader was not present in the delegation
hierarchy.
Preferred classes introduces the "class boomerang" problem to
RMI class loading. A class boomerang occurs when a class which
is marked preferred is accessible from the codebase of a
virtual machine (VM) and is loaded by that VM. Since the VM
has a copy of the class in its own resources and the class is
"returning to its origin" the class should not be preferred.
If the class is preferred, it will be loaded in a class loader
for the local codebase annotation. As a result, the class'
type will not be compatible with types defined from the local
definition of the class file in the relevant VM.
A boomerang can also occur when a new child loader for a URL
path is created but an ancestor of the new class loader has the
same URL path as the new class loader. In such cases the new
class loader should not be created. The incoming class should
be loaded from the origin ancestor instead.
A simple example of a class boomerang occurs when when a VM
makes a remote method call to itself. Suppose an object whose
class was loaded locally in that VM and is preferred in the
codebase of the VM is passed in the call. When the VM receives
its own call, an instance of the unmarshalled parameter will
not be assignable to instances that were defined by local
classes (i.e. never unmarshalled).
In order to work around class boomerang problems, the preferred
class provider lookupLoader algorithm is different from the
analogous algorithm in LoaderHandler. To avoid boomerangs, the
lookupLoader method of this class attempts to locate the
"origin" class loader of an incoming class in a remote method
call. Loading classes from their origin loader instead of in a
preferred circumvents type compatibility conflicts.
To find origin loaders, the lookupLoader method calls
findOriginLoader() before locating or creating new
PreferredClassLoaders. An origin loader is found by searching
up the delegation hierarchy above the parent (context) class
loader for a loader that has an export path which matches the
parameter path.private ClassLoader findOriginLoader0(Uri[] pathURLs, ClassLoader parent)
private ClassLoader lookupLoader(Uri[] uris, URL[] urls, ClassLoader parent) throws MalformedURLException
MalformedURLException
protected ClassLoader createClassLoader(URL[] urls, ClassLoader parent, boolean requireDlPerm)
PreferredClassProvider
to use to load classes from
the specified path of URLs with the specified delegation
parent.
PreferredClassProvider
implements this method
as follows:
This method creates a new instance of PreferredClassLoader
that loads classes and resources from
urls
, delegates to parent
, and
enforces DownloadPermission
if
requireDlPerm
is true
. The created
loader uses a restricted security context to ensure that the
URL retrieval operations undertaken by the loader cannot
exercise a permission that is not implied by the permissions
necessary to access the loader as a codebase loader for the
specified path of URLs.
urls
- the path of URLs to load classes and resources fromparent
- the parent class loader for delegationrequireDlPerm
- if true
, the loader must only
define classes with a CodeSource
that is granted
DownloadPermission
private static ClassLoader getClassLoader(Class c)
Copyright 2007-2013, multiple authors.
Licensed under the Apache License, Version 2.0, see the NOTICE file for attributions.