Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
/*
* Copyright (c) 2007, 2008, 2009, 2010, 2011, 2012, 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020 David Berkman
*
* This file is part of the SmallMind Code Project.
*
* The SmallMind Code Project is free software, you can redistribute
* it and/or modify it under either, at your discretion...
*
* 1) The terms of GNU Affero General Public License as published by the
* Free Software Foundation, either version 3 of the License, or (at
* your option) any later version.
*
* ...or...
*
* 2) The terms of the Apache License, Version 2.0.
*
* The SmallMind Code Project 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 or Apache License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* and the Apache License along with the SmallMind Code Project. If not, see
* or .
*
* Additional permission under the GNU Affero GPL version 3 section 7
* ------------------------------------------------------------------
* If you modify this Program, or any covered work, by linking or
* combining it with other code, such other code is not for that reason
* alone subject to any of the requirements of the GNU Affero GPL
* version 3.
*/
package org.smallmind.spark.singularity.boot;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.security.AllPermission;
import java.security.CodeSource;
import java.security.PermissionCollection;
import java.security.ProtectionDomain;
import java.security.cert.Certificate;
import java.util.Collections;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.HashSet;
import java.util.LinkedList;
import java.util.Map;
import java.util.NoSuchElementException;
import java.util.jar.Attributes;
import java.util.jar.JarEntry;
import java.util.jar.JarInputStream;
import java.util.jar.Manifest;
public class SingularityClassLoader extends ClassLoader {
private static final PermissionCollection ALL_PERMISSION_COLLECTION;
private static final String[] INOPERABLE_NAMESPACES = new String[] {"javax.xml.", "org.xml.", "org.w3c."};
private static final String[] OPERABLE_NAMESPACES = new String[] {"javax.xml.bind."};
private final HashMap urlMap = new HashMap<>();
private final HashSet packageSet = new HashSet<>();
private final URL sealBase;
private final String specificationTitle;
private final String specificationVersion;
private final String specificationVendor;
private final String implementationTitle;
private final String implementationVersion;
private final String implementationVendor;
static {
AllPermission allPermission = new AllPermission();
ALL_PERMISSION_COLLECTION = allPermission.newPermissionCollection();
ALL_PERMISSION_COLLECTION.add(allPermission);
ClassLoader.registerAsParallelCapable();
URL.setURLStreamHandlerFactory(new SingularityJarURLStreamHandlerFactory());
}
public SingularityClassLoader (ClassLoader parent, Manifest manifest, URL jarURL, JarInputStream jarInputStream)
throws IOException, ClassNotFoundException {
super(parent);
SingularityIndex singularityIndex = null;
Attributes mainAttributes = manifest.getMainAttributes();
JarEntry jarEntry;
String sealed;
while ((jarEntry = jarInputStream.getNextJarEntry()) != null) {
if (!jarEntry.isDirectory()) {
if (jarEntry.getName().equals("META-INF/singularity/index/singularity.idx")) {
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
int singleByte;
while ((singleByte = jarInputStream.read()) >= 0) {
byteArrayOutputStream.write(singleByte);
}
byteArrayOutputStream.close();
try (ObjectInputStream objectInputStream = new ObjectInputStream(new ByteArrayInputStream(byteArrayOutputStream.toByteArray()))) {
singularityIndex = (SingularityIndex)objectInputStream.readObject();
}
break;
}
}
}
if (singularityIndex == null) {
throw new IOException("Missing singularity index");
}
for (SingularityIndex.URLEntry urlEntry : singularityIndex.getJarURLEntryIterable(jarURL.toExternalForm())) {
urlMap.put(urlEntry.getEntryName(), urlEntry.getEntryURL());
}
for (SingularityIndex.URLEntry urlEntry : singularityIndex.getSingularityURLEntryIterable(jarURL.toExternalForm())) {
urlMap.put(urlEntry.getEntryName(), urlEntry.getEntryURL());
}
specificationTitle = mainAttributes.getValue(Attributes.Name.SPECIFICATION_TITLE);
specificationVersion = mainAttributes.getValue(Attributes.Name.SPECIFICATION_VERSION);
specificationVendor = mainAttributes.getValue(Attributes.Name.SPECIFICATION_VENDOR);
implementationTitle = mainAttributes.getValue(Attributes.Name.IMPLEMENTATION_TITLE);
implementationVersion = mainAttributes.getValue(Attributes.Name.IMPLEMENTATION_VERSION);
implementationVendor = mainAttributes.getValue(Attributes.Name.IMPLEMENTATION_VENDOR);
if ((sealed = mainAttributes.getValue(Attributes.Name.SEALED)) != null) {
if (Boolean.parseBoolean(sealed)) {
sealBase = jarURL;
} else {
sealBase = null;
}
} else {
sealBase = null;
}
}
@Override
public synchronized Class> loadClass (String name, boolean resolve)
throws ClassNotFoundException {
Class> singularityClass;
if ((singularityClass = findLoadedClass(name)) == null) {
try {
singularityClass = findClass(name);
} catch (ClassNotFoundException c) {
if (getParent() != null) {
singularityClass = getParent().loadClass(name);
} else {
singularityClass = findSystemClass(name);
}
}
}
if (resolve) {
resolveClass(singularityClass);
}
return singularityClass;
}
@Override
public synchronized Class> findClass (String name)
throws ClassNotFoundException {
if (isOperableNamespace(name)) {
URL classURL;
URL codeSourceUrl;
if ((classURL = urlMap.get(name.replace('.', '/') + ".class")) != null) {
try {
String classURLExternalForm = classURL.toExternalForm();
switch (classURL.getProtocol()) {
case "jar":
int jarBangSlashIndex;
if ((jarBangSlashIndex = classURLExternalForm.indexOf("!/")) < 0) {
codeSourceUrl = new URL(classURLExternalForm + "!/");
} else {
codeSourceUrl = new URL(classURLExternalForm.substring(0, jarBangSlashIndex + 2));
}
break;
case "singularity":
// javax.crypto.JarVerifier will encase this URL in 'jar:!/ (it incorrectly assumes any protocol not 'jar:' is 'file:'),
// which then gets stripped again by JarUrlConnection, which will let SingularityJarURLConnection respond correctly...
codeSourceUrl = new URL(classURLExternalForm.substring(0, classURLExternalForm.indexOf("!/")));
break;
default:
throw new MalformedURLException("Unknown class url protocol(" + classURL.getProtocol() + ")");
}
CodeSource codeSource = new CodeSource(codeSourceUrl, (Certificate[])null);
ProtectionDomain protectionDomain = new ProtectionDomain(codeSource, ALL_PERMISSION_COLLECTION, this, null);
InputStream classInputStream;
byte[] classData;
classInputStream = classURL.openStream();
classData = getClassData(classInputStream);
classInputStream.close();
definePackage(name);
return defineClass(name, classData, 0, classData.length, protectionDomain);
} catch (Exception exception) {
throw new ClassNotFoundException("Exception encountered while attempting to define class (" + name + ")", exception);
}
}
}
throw new ClassNotFoundException(name);
}
private boolean isOperableNamespace (String name) {
for (String operableNamespace : OPERABLE_NAMESPACES) {
if (name.startsWith(operableNamespace)) {
return true;
}
}
for (String inoperableNamespace : INOPERABLE_NAMESPACES) {
if (name.startsWith(inoperableNamespace)) {
return false;
}
}
return true;
}
private void definePackage (String name) {
String packageName;
int lastDotPos = name.lastIndexOf('.');
packageName = name.substring(0, lastDotPos);
if (packageSet.add(packageName)) {
definePackage(packageName, specificationTitle, specificationVersion, specificationVendor, implementationTitle, implementationVersion, implementationVendor, sealBase);
}
}
private byte[] getClassData (InputStream classInputStream)
throws IOException {
ByteArrayOutputStream classDataOutputStream = new ByteArrayOutputStream();
int singleByte;
while ((singleByte = classInputStream.read()) >= 0) {
classDataOutputStream.write(singleByte);
}
return classDataOutputStream.toByteArray();
}
@Override
public URL findResource (String name) {
if ((name == null) || name.isEmpty()) {
return null;
} else {
return urlMap.get((name.charAt(0) == '/') ? name.substring(1) : name);
}
}
@Override
protected Enumeration findResources (String name) {
if ((name == null) || name.isEmpty()) {
return Collections.emptyEnumeration();
} else if (!name.endsWith("/")) {
URL url;
if ((url = findResource(name)) == null) {
return Collections.emptyEnumeration();
}
return new SingleEnumeration<>(url);
} else {
LinkedList urlList = new LinkedList<>();
for (Map.Entry resourceEntry : urlMap.entrySet()) {
if (resourceEntry.getKey().startsWith(name) && (!resourceEntry.getKey().endsWith("/"))) {
urlList.add(resourceEntry.getValue());
}
}
if (urlList.isEmpty()) {
return Collections.emptyEnumeration();
} else {
URL[] urls = new URL[urlList.size()];
urlList.toArray(urls);
return new ArrayEnumeration<>(urls);
}
}
}
private static class SingleEnumeration implements Enumeration {
private final T value;
private boolean used = false;
private SingleEnumeration (T value) {
this.value = value;
}
@Override
public synchronized boolean hasMoreElements () {
return !used;
}
@Override
public synchronized T nextElement () {
if (used) {
throw new NoSuchElementException();
}
used = true;
return value;
}
}
private static class ArrayEnumeration implements Enumeration {
private final T[] values;
private int index = 0;
private ArrayEnumeration (T[] values) {
this.values = values;
}
@Override
public synchronized boolean hasMoreElements () {
return index < values.length;
}
@Override
public synchronized T nextElement () {
return values[index++];
}
}
}