blob: bd7e608f3132f5d25403ede1446248304fea9394 [file] [log] [blame]
// Copyright (C) 2018 The Android Open Source Project
//
// Licensed 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 com.google.gerrit.server.logging;
import static com.google.common.flogger.LazyArgs.lazy;
import com.google.auto.value.AutoValue;
import com.google.common.collect.ImmutableList;
import com.google.common.flogger.LazyArg;
import java.util.Optional;
/**
* Utility to compute the caller of a method.
*
* <p>In the logs we see for each entry from where it was triggered (class/method/line) but in case
* the logging is done in a utility method or inside of a module this doesn't tell us from where the
* action was actually triggered. To get this information we could included the stacktrace into the
* logs (by calling {@link
* com.google.common.flogger.LoggingApi#withStackTrace(com.google.common.flogger.StackSize)} but
* sometimes there are too many uninteresting stacks so that this would blow up the logs too much.
* In this case CallerFinder can be used to find the first interesting caller from the current
* stacktrace by specifying the class that interesting callers invoke as target.
*
* <p>Example:
*
* <p>Index queries are executed by the {@code query(List<String>, List<Predicate<T>>)} method in
* {@link com.google.gerrit.index.query.QueryProcessor}. At this place the index query is logged but
* from the log we want to see which code triggered this index query.
*
* <p>E.g. the stacktrace could look like this:
*
* <pre>
* GroupQueryProcessor(QueryProcessor<T>).query(List<String>, List<Predicate<T>>) line: 216
* GroupQueryProcessor(QueryProcessor<T>).query(List<Predicate<T>>) line: 188
* GroupQueryProcessor(QueryProcessor<T>).query(Predicate<T>) line: 171
* InternalGroupQuery(InternalQuery<T>).query(Predicate<T>) line: 81
* InternalGroupQuery.getOnlyGroup(Predicate<InternalGroup>, String) line: 67
* InternalGroupQuery.byName(NameKey) line: 50
* GroupCacheImpl$ByNameLoader.load(String) line: 166
* GroupCacheImpl$ByNameLoader.load(Object) line: 1
* LocalCache$LoadingValueReference<K,V>.loadFuture(K, CacheLoader<? super K,V>) line: 3527
* ...
* </pre>
*
* <p>The first interesting caller is {@code GroupCacheImpl$ByNameLoader.load(String) line: 166}. To
* find this caller from the stacktrace we could specify {@link
* com.google.gerrit.server.query.group.InternalGroupQuery} as a target since we know that all
* internal group queries go through this class:
*
* <pre>
* CallerFinder.builder()
* .addTarget(InternalGroupQuery.class)
* .build();
* </pre>
*
* <p>Since in some places {@link com.google.gerrit.server.query.group.GroupQueryProcessor} may also
* be used directly we can add it as a secondary target to catch these callers as well:
*
* <pre>
* CallerFinder.builder()
* .addTarget(InternalGroupQuery.class)
* .addTarget(GroupQueryProcessor.class)
* .build();
* </pre>
*
* <p>However since {@link com.google.gerrit.index.query.QueryProcessor} is also responsible to
* execute other index queries (for changes, accounts, projects) we would need to add the classes
* for them as targets too. Since there are common base classes we can simply specify the base
* classes and request matching of subclasses:
*
* <pre>
* CallerFinder.builder()
* .addTarget(InternalQuery.class)
* .addTarget(QueryProcessor.class)
* .matchSubClasses(true)
* .build();
* </pre>
*
* <p>Another special case is if the entry point is always an inner class of a known interface. E.g.
* {@link com.google.gerrit.server.permissions.PermissionBackend} is the entry point for all
* permission checks but they are done through inner classes, e.g. {@link
* com.google.gerrit.server.permissions.PermissionBackend.ForProject}. In this case matching of
* inner classes must be enabled as well:
*
* <pre>
* CallerFinder.builder()
* .addTarget(PermissionBackend.class)
* .matchSubClasses(true)
* .matchInnerClasses(true)
* .build();
* </pre>
*
* <p>Finding the interesting caller requires specifying the entry point class as target. This may
* easily break when code is refactored and hence should be used only with care. It's recommended to
* use this only when the corresponding code is relatively stable and logging the caller information
* brings some significant benefit.
*
* <p>Based on {@link com.google.common.flogger.util.CallerFinder}.
*/
@AutoValue
public abstract class CallerFinder {
public static Builder builder() {
return new AutoValue_CallerFinder.Builder()
.matchSubClasses(false)
.matchInnerClasses(false)
.skip(0);
}
/**
* The target classes for which the caller should be found, in the order in which they should be
* checked.
*
* @return the target classes for which the caller should be found
*/
public abstract ImmutableList<Class<?>> targets();
/**
* Whether inner classes should be matched.
*
* @return whether inner classes should be matched
*/
public abstract boolean matchSubClasses();
/**
* Whether sub classes of the target classes should be matched.
*
* @return whether sub classes of the target classes should be matched
*/
public abstract boolean matchInnerClasses();
/**
* The minimum number of calls known to have occurred between the first call to the target class
* and the call of {@link #findCallerLazy()}. If in doubt, specify zero here to avoid accidentally
* skipping past the caller.
*
* @return the number of stack elements to skip when computing the caller
*/
public abstract int skip();
/**
* Packages that should be ignored and not be considered as caller once a target has been found.
*
* @return the ignored packages
*/
public abstract ImmutableList<String> ignoredPackages();
/**
* Classes that should be ignored and not be considered as caller once a target has been found.
*
* @return the qualified names of the ignored classes
*/
public abstract ImmutableList<String> ignoredClasses();
@AutoValue.Builder
public abstract static class Builder {
abstract ImmutableList.Builder<Class<?>> targetsBuilder();
public Builder addTarget(Class<?> target) {
targetsBuilder().add(target);
return this;
}
public abstract Builder matchSubClasses(boolean matchSubClasses);
public abstract Builder matchInnerClasses(boolean matchInnerClasses);
public abstract Builder skip(int skip);
abstract ImmutableList.Builder<String> ignoredPackagesBuilder();
public Builder addIgnoredPackage(String ignoredPackage) {
ignoredPackagesBuilder().add(ignoredPackage);
return this;
}
abstract ImmutableList.Builder<String> ignoredClassesBuilder();
public Builder addIgnoredClass(Class<?> ignoredClass) {
ignoredClassesBuilder().add(ignoredClass.getName());
return this;
}
public abstract CallerFinder build();
}
public LazyArg<String> findCallerLazy() {
return lazy(
() ->
targets().stream()
.map(t -> findCallerOf(t, skip() + 1))
.filter(Optional::isPresent)
.findFirst()
.map(Optional::get)
.orElse("unknown"));
}
private Optional<String> findCallerOf(Class<?> target, int skip) {
// Skip one additional stack frame because we create the Throwable inside this method, not at
// the point that this method was invoked.
skip++;
StackTraceElement[] stack = new Throwable().getStackTrace();
// Note: To avoid having to reflect the getStackTraceDepth() method as well, we assume that we
// will find the caller on the stack and simply catch an exception if we fail (which should
// hardly ever happen).
boolean foundCaller = false;
try {
for (int index = skip; ; index++) {
StackTraceElement element = stack[index];
if (isCaller(target, element.getClassName(), matchSubClasses())) {
foundCaller = true;
} else if (foundCaller
&& !ignoredPackages().contains(getPackageName(element))
&& !ignoredClasses().contains(element.getClassName())) {
return Optional.of(element.toString());
}
}
} catch (Exception e) {
// This should only happen if a) the caller was not found on the stack
// (IndexOutOfBoundsException) b) a class that is mentioned in the stack was not found
// (ClassNotFoundException), however we don't want anything to be thrown from here.
return Optional.empty();
}
}
private static String getPackageName(StackTraceElement element) {
String className = element.getClassName();
return className.substring(0, className.lastIndexOf("."));
}
private boolean isCaller(Class<?> target, String className, boolean matchSubClasses)
throws ClassNotFoundException {
if (matchSubClasses) {
Class<?> clazz = Class.forName(className);
while (clazz != null) {
if (Object.class.getName().equals(clazz.getName())) {
break;
}
if (isCaller(target, clazz.getName(), false)) {
return true;
}
clazz = clazz.getSuperclass();
}
}
if (matchInnerClasses()) {
int i = className.indexOf('$');
if (i > 0) {
className = className.substring(0, i);
}
}
if (target.getName().equals(className)) {
return true;
}
return false;
}
}