All Downloads are FREE. Search and download functionalities are using the official Maven repository.

net.sf.jasperreports.engine.util.JRClassLoader Maven / Gradle / Ivy

There is a newer version: 6.21.3
Show newest version
/*
 * JasperReports - Free Java Reporting Library.
 * Copyright (C) 2001 - 2019 TIBCO Software Inc. All rights reserved.
 * http://www.jaspersoft.com
 *
 * Unless you have purchased a commercial license agreement from Jaspersoft,
 * the following license terms apply:
 *
 * This program is part of JasperReports.
 *
 * JasperReports 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.
 *
 * JasperReports 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 JasperReports. If not, see .
 */
package net.sf.jasperreports.engine.util;

import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.security.ProtectionDomain;
import java.util.HashMap;
import java.util.Map;

import net.sf.jasperreports.engine.JRRuntimeException;

/**
 * @author Teodor Danciu ([email protected])
 */
public class JRClassLoader extends ClassLoader
{
	
	private static final Map PRIMITIVE_COMPONENT_ENCODING;
	static
	{
		PRIMITIVE_COMPONENT_ENCODING = new HashMap<>();
		//taken from https://docs.oracle.com/javase/8/docs/api/java/lang/Class.html#getName--
		PRIMITIVE_COMPONENT_ENCODING.put(Boolean.TYPE.getName(), "Z");
		PRIMITIVE_COMPONENT_ENCODING.put(Byte.TYPE.getName(), "B");
		PRIMITIVE_COMPONENT_ENCODING.put(Character.TYPE.getName(), "C");
		PRIMITIVE_COMPONENT_ENCODING.put(Double.TYPE.getName(), "D");
		PRIMITIVE_COMPONENT_ENCODING.put(Float.TYPE.getName(), "F");
		PRIMITIVE_COMPONENT_ENCODING.put(Integer.TYPE.getName(), "I");
		PRIMITIVE_COMPONENT_ENCODING.put(Long.TYPE.getName(), "J");
		PRIMITIVE_COMPONENT_ENCODING.put(Short.TYPE.getName(), "S");
	}

	private static ProtectionDomainFactory protectionDomainFactory;
	
	public static synchronized ProtectionDomainFactory getProtectionDomainFactory()
	{
		if (protectionDomainFactory == null)
		{
			protectionDomainFactory = new SingleProtectionDomainFactory(JRClassLoader.class.getProtectionDomain());
		}

		return protectionDomainFactory;
	}
	
	/**
	 * Sets the protection to be used for classes loaded via
	 * the {@link #loadClassFromBytes(String, byte[]) loadClassFromBytes} method.
	 *
	 * By default, the protection domain of this class is used for the loaded classes. 
	 * 
	 * @param protectionDomain the protection domain to be used
	 * @see #loadClassFromBytes(String, byte[])
	 */
	public static void setProtectionDomain(ProtectionDomain protectionDomain)
	{
		SingleProtectionDomainFactory factory = new SingleProtectionDomainFactory(protectionDomain);
		setProtectionDomainFactory(factory);
	}
	
	/**
	 * Sets a protection domain factory to be used for creating protection domains
	 * for the classes loaded by instances of this class.
	 * 

* For every instance of this class, * {@link ProtectionDomainFactory#getProtectionDomain(ClassLoader) getProtectionDomain} is called * and the resulting protection domain is used when loading classes through the newly created * classloader. * * @param protectionDomainFactory the protection domain factory. * @see ProtectionDomainFactory#getProtectionDomain(ClassLoader) */ public static void setProtectionDomainFactory(ProtectionDomainFactory protectionDomainFactory) { JRClassLoader.protectionDomainFactory = protectionDomainFactory; } private ProtectionDomain protectionDomain; private ClassLoaderFilter classLoaderFilter; /** * */ protected JRClassLoader() { super(); } protected JRClassLoader(ClassLoaderFilter classLoaderFilter) { super(); this.classLoaderFilter = classLoaderFilter; } /** * */ protected JRClassLoader(ClassLoader parent) { super(parent); } protected JRClassLoader(ClassLoader parent, ClassLoaderFilter classLoaderFilter) { super(parent); this.classLoaderFilter = classLoaderFilter; } /** * */ public static Class loadClassForName(String className) throws ClassNotFoundException { Class clazz = null; String classRealName = className; ClassNotFoundException initialEx = null; try { clazz = loadClassForRealName(classRealName); } catch (ClassNotFoundException e) { initialEx = e; } int lastDotIndex = 0; while (clazz == null && (lastDotIndex = classRealName.lastIndexOf('.')) > 0) { classRealName = classRealName.substring(0, lastDotIndex) + "$" + classRealName.substring(lastDotIndex + 1); try { clazz = loadClassForRealName(classRealName); } catch (ClassNotFoundException e) { } } if (clazz == null) { throw initialEx; } return clazz; } /** * */ public static Class loadClassForRealName(String className) throws ClassNotFoundException { Class clazz = null; ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); if (classLoader != null) { try { clazz = Class.forName(className, true, classLoader); } catch (ClassNotFoundException e) { //if (log.isWarnEnabled()) // log.warn("Failure using Thread.currentThread().getContextClassLoader() in JRClassLoader class. Using JRClassLoader.class.getClassLoader() instead."); } } if (clazz == null) { classLoader = JRClassLoader.class.getClassLoader(); if (classLoader == null) { clazz = Class.forName(className); } else { clazz = Class.forName(className, true, classLoader); } } return clazz; } /** * */ public static Class loadClassFromFile(String className, File file) throws IOException { Class clazz = null; ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); if (classLoader != null) { try { clazz = (new JRClassLoader(classLoader)) .loadClass(className, file); } catch(NoClassDefFoundError e) { //if (log.isWarnEnabled()) // log.warn("Failure using Thread.currentThread().getContextClassLoader() in JRClassLoader class. Using JRClassLoader.class.getClassLoader() instead."); } } if (clazz == null) { classLoader = JRClassLoader.class.getClassLoader(); if (classLoader == null) { clazz = (new JRClassLoader()) .loadClass(className, file); } else { clazz = (new JRClassLoader(classLoader)) .loadClass(className, file); } } return clazz; } /** * */ public static Class loadClassFromBytes(String className, byte[] bytecodes) { return loadClassFromBytes(null, className, bytecodes); } public static Class loadClassFromBytes(ClassLoaderFilter classLoaderFilter, String className, byte[] bytecodes) { Class clazz = null; ClassLoader classLoader = Thread.currentThread().getContextClassLoader(); if (classLoader != null) { try { clazz = (new JRClassLoader(classLoader, classLoaderFilter)) .loadClass(className, bytecodes); } catch(NoClassDefFoundError e) { //if (log.isWarnEnabled()) // log.warn("Failure using Thread.currentThread().getContextClassLoader() in JRClassLoader class. Using JRClassLoader.class.getClassLoader() instead."); } } if (clazz == null) { classLoader = JRClassLoader.class.getClassLoader(); if (classLoader == null) { clazz = (new JRClassLoader(classLoaderFilter)) .loadClass(className, bytecodes); } else { clazz = (new JRClassLoader(classLoader, classLoaderFilter)) .loadClass(className, bytecodes); } } return clazz; } /** * */ protected Class loadClass(String className, File file) throws IOException { FileInputStream fis = null; ByteArrayOutputStream baos = null; byte[] bytecodes = new byte[10000]; int ln = 0; try { fis = new FileInputStream(file); baos = new ByteArrayOutputStream(); while ( (ln = fis.read(bytecodes)) > 0 ) { baos.write(bytecodes, 0, ln); } baos.flush(); } finally { if (baos != null) { try { baos.close(); } catch(IOException e) { } } if (fis != null) { try { fis.close(); } catch(IOException e) { } } } return loadClass(className, baos.toByteArray()); } protected synchronized ProtectionDomain getProtectionDomain() { if (protectionDomain == null) { protectionDomain = getProtectionDomainFactory().getProtectionDomain(this); } return protectionDomain; } /** * */ protected Class loadClass(String className, byte[] bytecodes) { Class clazz = null; clazz = defineClass( className, bytecodes, 0, bytecodes.length, getProtectionDomain() ); return clazz; } @Override protected Class loadClass(String name, boolean resolve) throws ClassNotFoundException, JRRuntimeException { if (classLoaderFilter != null) { classLoaderFilter.checkClassVisibility(name); } return super.loadClass(name, resolve); } /** * */ public static String getClassRealName(String className) { if (className == null) { return null; } int ltPos = className.indexOf('<'); if (ltPos > 0) { int gtPos = className.lastIndexOf('>'); if (gtPos > ltPos) { className = className.substring(0, ltPos) + className.substring(gtPos); } } int arrayDimension = 0; int classNameEnd = className.length(); int index = 0; int pos = 0; while (index < classNameEnd && (pos = className.indexOf('[', index)) >= 0) { if (index == 0) { classNameEnd = pos; } index = pos; arrayDimension++; } if (arrayDimension > 0) { StringBuilder sb = new StringBuilder(); for(int i = 0; i < arrayDimension; i++) { sb.append('['); } String componentClass = className.substring(0, classNameEnd); String primitiveEncoding = PRIMITIVE_COMPONENT_ENCODING.get(componentClass); if (primitiveEncoding == null) { //non primitive component sb.append('L'); sb.append(componentClass); sb.append(';'); } else { sb.append(primitiveEncoding); } return sb.toString(); } return className; } }





© 2015 - 2024 Weber Informatics LLC | Privacy Policy