blob: d27fb9cb1d20c538ee56f85501052f73cdfda727 [file] [log] [blame]
/*
* 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.logging.log4j.util;
import org.apache.logging.log4j.status.StatusLogger;
import java.util.NoSuchElementException;
import java.util.Stack;
import java.util.function.Predicate;
/**
* <em>Consider this class private.</em> Provides various methods to determine the caller class. <h3>Background</h3>
*/
public final class StackLocatorUtil {
private static StackLocator stackLocator = null;
private static volatile boolean errorLogged;
static {
stackLocator = StackLocator.getInstance();
}
private StackLocatorUtil() {
}
// TODO: return Object.class instead of null (though it will have a null ClassLoader)
// (MS) I believe this would work without any modifications elsewhere, but I could be wrong
// migrated from ReflectiveCallerClassUtility
@PerformanceSensitive
public static Class<?> getCallerClass(final int depth) {
return stackLocator.getCallerClass(depth + 1);
}
public static StackTraceElement getStackTraceElement(final int depth) {
return stackLocator.getStackTraceElement(depth + 1);
}
/**
* Equivalent to {@link #getCallerClass(String, String)} with an empty {@code pkg}.
*/
// migrated from ClassLoaderContextSelector
@PerformanceSensitive
public static Class<?> getCallerClass(final String fqcn) {
return getCallerClass(fqcn, Strings.EMPTY);
}
/**
* Search for a calling class.
*
* @param fqcn Root class name whose caller to search for.
* @param pkg Package name prefix that must be matched after the {@code fqcn} has been found.
* @return The caller class that was matched, or null if one could not be located.
*/
@PerformanceSensitive
public static Class<?> getCallerClass(final String fqcn, final String pkg) {
return stackLocator.getCallerClass(fqcn, pkg);
}
/**
* Search for a calling class.
*
* @param sentinelClass Sentinel class at which to begin searching
* @param callerPredicate Predicate checked after the sentinelClass is found
* @return the first matching class after <code>sentinelClass</code> is found.
*/
@PerformanceSensitive
public static Class<?> getCallerClass(final Class<?> sentinelClass, final Predicate<Class<?>> callerPredicate) {
return stackLocator.getCallerClass(sentinelClass, callerPredicate);
}
// added for use in LoggerAdapter implementations mainly
@PerformanceSensitive
public static Class<?> getCallerClass(final Class<?> anchor) {
return stackLocator.getCallerClass(anchor);
}
// migrated from ThrowableProxy
@PerformanceSensitive
public static Stack<Class<?>> getCurrentStackTrace() {
return stackLocator.getCurrentStackTrace();
}
public static StackTraceElement calcLocation(final String fqcnOfLogger) {
try {
return stackLocator.calcLocation(fqcnOfLogger);
} catch (final NoSuchElementException ex) {
if (!errorLogged) {
errorLogged = true;
StatusLogger.getLogger().warn("Unable to locate stack trace element for {}", fqcnOfLogger, ex);
}
return null;
}
}
}