An open API service indexing awesome lists of open source software.

https://github.com/sonarsource/sonar-classloader

Toolbox for Java classloaders
https://github.com/sonarsource/sonar-classloader

Last synced: 6 months ago
JSON representation

Toolbox for Java classloaders

Awesome Lists containing this project

README

          

# Toolbox for Java Classloaders

Sonar Classloader is a toolbox for creating Java 7+ classloaders. It's inspired from projects [Codehaus Classworlds][classworlds] and [Plexus Classworlds][plexus]. It is not coupled with SonarQube and can be used without any third-party dependencies.

This library is available under GNU LGPLv3.

## Maven Dependency


org.sonarsource.classloader
sonar-classloader
1.1

## Usage

#### Build classloader

Create a classloader based on system classloader.

```java
ClassloaderBuilder builder = new ClassloaderBuilder();
Map classloaders = builder
.newClassloader("the-cl")
.addURL("the-cl", jarFile)
.addURL("the-cl", directory)
.build();

// this classloader can load only JRE and the resources contained in jarFile and directory.
ClassLoader c = classloaders.get("the-cl");
```

It's also possible to create a classloader based on another one:

```java
ClassloaderBuilder builder = new ClassloaderBuilder();
Map classloaders = builder
.newClassloader("the-cl", otherClassloader)
.addURL("the-cl", jarFile)
.build();

// this classloader can load the resources of JRE, jarFile and otherClassloader.
ClassLoader cl1 = classloaders.get("cl1");
```

#### Hierarchy of classloaders

```java
ClassloaderBuilder builder = new ClassloaderBuilder();
Map classloaders = builder
.newClassloader("the-parent")
.addURL("the-parent", parentJar)

.newClassloader("the-child")
.addURL("the-child", childJar)
.setParent("the-child", "the-parent", Mask.ALL)

.newClassloader("the-grand-child")
.setParent("the-grand-child", "the-child", Mask.ALL)
// can be parent-first or self-first ordering strategy. Default is parent-first.
.setLoadingOrder("the-grand-child", LoadingOrder.SELF_FIRST)

.build();
ClassLoader parent = classloaders.get("the-parent");
ClassLoader child = classloaders.get("the-child");
ClassLoader grandChild = classloaders.get("the-grand-child");
```

Importing classes and resources from sibling classloaders allows to convert the standard tree of classloaders to a graph a classloaders.

```java
ClassloaderBuilder builder = new ClassloaderBuilder();
// build 4 classloaders. "the-child" searches for resources from sibling1, sibling2, the-parent then itself.
Map classloaders = builder
.newClassloader("the-parent")
.newClassloader("sibling1")
.newClassloader("sibling2")
.newClassloader("the-child")
.setParent("the-child", "the-parent", Mask.ALL)
.addSibling("the-child", "sibling1", Mask.ALL)
.addSibling("the-child", "sibling2", Mask.ALL)
.build();
```

#### Masks

A mask restricts access of a classloader to some resources and classes. By default there are no restrictions.

A mask is based on inclusion and exclusion patterns. Format is file path separated by slashes, for example "org/foo/Bar.class" or "org/foo/config.xml". Wildcard patterns are not supported. Directories must end with slash, for example "org/foo/" for excluding package org.foo and all its sub-packages.

Masks can be defined on parent-child and sibling relations.

```java
// all the resources/classes of classloader 'b' are visible from 'a', except org/foo/** resources.
ClassloaderBuilder builder = new ClassloaderBuilder();
Map classloaders = builder
.newClassloader("a")
.newClassloader("b")
.addSibling("a", "b", Mask.builder().exclude("org/foo/").build())
.build();
```

When the same patterns are defined multiple times for each usage of a classloader, then the patterns can be declared once on the targetted classloader.

```java
// classloader 'a' exports only the resources/classes "org/a/api/" but not other internal classes.
ClassloaderBuilder builder = new ClassloaderBuilder();
Map classloaders = builder
.newClassloader("a")
.setExportMask(Mask.builder().include("org/a/api/").build())
.newClassloader("b")
.newClassloader("c")
.addSibling("b", "a", Mask.ALL)
.addSibling("c", "a", Mask.ALL)
.build();
```

This is equivalent of:

```java
ClassloaderBuilder builder = new ClassloaderBuilder();
Map classloaders = builder
.newClassloader("a")
.newClassloader("b")
.newClassloader("c")
.addSibling("b", "a", Mask.builder().include("org/a/api/").build())
.addSibling("c", "a", Mask.builder().include("org/a/api/").build())
.build();
```

## License

Copyright (C) 2015 SonarSource

This program is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 3 of the License, or (at your option) any later version.

This program 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
Lesser General Public License for more details.

You should have received a copy of the GNU Lesser General Public
License along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02

[classworlds]: http://classworlds.codehaus.org
[plexus]: https://github.com/sonatype/plexus-classworlds