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

org.apache.log4j.helpers.Loader Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You under the Apache License, Version 2.0
 * (the "License"); you may not use this file except in compliance with
 * the License.  You may obtain a copy of the License at
 *
 *      http://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */

package org.apache.log4j.helpers;

import java.net.URL;
import java.lang.reflect.Method;
import java.lang.reflect.InvocationTargetException;
import java.io.InterruptedIOException;

import org.osgi.framework.Bundle;
import org.osgi.framework.FrameworkUtil;

/**
 * Load resources (or images) from various sources.
 *
 * @author Ceki Gülcü
 */

public class Loader {

    static final String TSTR = "Caught Exception while in Loader.getResource. This may be innocuous.";

    static private boolean ignoreTCL = false;

    /**
     * Get a resource by delegating to getResource(String).
     *
     * @param resource resource name
     * @param clazz    class, ignored.
     * @return URL to resource or null.
     * @deprecated as of 1.2.
     */
    public static URL getResource(String resource, Class clazz) {
        return getResource(resource);
    }

    /**
     * This method will search for resource in different places. The search order is as follows:
     * 
    *
  1. Search for resource using the thread context class loader, * unless "log4j.ignoreTCL" system property was set to true.

    *
  2. *
  3. Search for resource using the class * loader that loaded this class (Loader).

    *

  4. *
  5. In Pax Logging, {@link org.osgi.framework.Bundle#getResource(String)} is first checked
  6. *
  7. Try one last time with * ClassLoader.getSystemResource(resource), that is is using the * system class loader. *
* *

Nota bene: In versions of reload4j 1.2.23 and earlier, the jaadoc documentation stated that * the thread context class loader was used but when running under JDK 9 and later this * was not actually the case. As of version 1.2.25, the javadoc corresponds to the original * intention as documented. *

* * * @param resource the resource to load */ static public URL getResource(String resource) { Bundle bundle = FrameworkUtil.getBundle(Loader.class); if (bundle != null) { LogLog.debug("Trying to find [" + resource + "] using " + bundle.getSymbolicName() + "/" + bundle.getVersion() + " bundle."); URL url = bundle.getResource(resource); if (url != null) { return url; } } try { // unless intsructed to ignore the TCL, try getting the resource using TCL, return if found. if(!ignoreTCL) { URL url0 = innerGetResource(resource, getTCL()); if(url0 != null) { return url0; } } // if we were instructed to ignore TCL or if no url was dound, try using the // class loader that loaded this class. URL url = innerGetResource(resource, Loader.class.getClassLoader()); if(url != null) { return url; } } catch (SecurityException t) { // can't be InterruptedException or InterruptedIOException // since not declared, must be error or RuntimeError. LogLog.warn(TSTR, t); } catch (InvocationTargetException e) { throw new RuntimeException(e); } catch (IllegalAccessException e) { throw new RuntimeException(e); } // Last ditch attempt: get the resource from the class path. It // may be the case that clazz was loaded by the Extentsion class // loader which the parent of the system class loader. Hence the // code below. LogLog.debug("Trying to find [" + resource + "] using ClassLoader.getSystemResource()."); return ClassLoader.getSystemResource(resource); } private static URL innerGetResource(String resource, ClassLoader classLoader) { if (classLoader != null) { LogLog.debug("Trying to find [" + resource + "] using " + classLoader + " class loader."); return classLoader.getResource(resource); } else { return null; } } /** * Are we running under JDK 1.x? * @deprecated with no replacement */ @Deprecated public static boolean isJava1() { return false; } /** * Get the Thread Context Loader which is a JDK 1.2 feature. If we are running under JDK 1.1 or anything else goes * wrong the method returns * null. */ private static ClassLoader getTCL() throws IllegalAccessException, InvocationTargetException { // Are we running on a JDK 1.2 or later system? Method method = null; try { method = Thread.class.getMethod("getContextClassLoader", null); } catch (NoSuchMethodException e) { // We are running on JDK 1.1 return null; } return (ClassLoader) method.invoke(Thread.currentThread(), null); } /** * Load the specified class using the {@linl Class#forName} method. * *

Nota bene: In versions of reload4j 1.2.23 and earlier, the documentation stated that * the thread context class loader was used to load the specified class but * when running under JDK 9 and later this was not actually the case. As of version 1.2.24, * the javadoc above matches the code as executed. *

* * In pax-logging, TCCL is always checked first, then {@link Bundle#loadClass(String)} * eventually {@link Class#forName(String)} is called. * * @param clazz the name of class to load */ static public Class loadClass(String clazz) throws ClassNotFoundException { if (Thread.currentThread().getContextClassLoader() != null) { try { return Thread.currentThread().getContextClassLoader().loadClass(clazz); } catch (Exception ignored) { } } Bundle bundle = FrameworkUtil.getBundle(Loader.class); if (bundle != null) { try { return bundle.loadClass(clazz); } catch (Exception ignored) { } } return Class.forName(clazz); } }