blob: ab5db023bf282bee6913ef759ba5913136a7caaf [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 com.google.common.collect.ImmutableSetMultimap;
import com.google.common.flogger.FluentLogger;
import java.util.concurrent.Callable;
/**
* Wrapper for a {@link Callable} that copies the {@link LoggingContext} from the current thread to
* the thread that executes the callable.
*
* <p>The state of the logging context that is copied to the thread that executes the callable is
* fixed at the creation time of this wrapper. If the callable is submitted to an executor and is
* executed later this means that changes that are done to the logging context in between creating
* and executing the callable do not apply.
*
* <p>See {@link LoggingContextAwareRunnable} for an example.
*
* @see LoggingContextAwareRunnable
*/
class LoggingContextAwareCallable<T> implements Callable<T> {
private static final FluentLogger logger = FluentLogger.forEnclosingClass();
private final Callable<T> callable;
private final Thread callingThread;
private final ImmutableSetMultimap<String, String> tags;
private final boolean forceLogging;
private final boolean performanceLogging;
private final MutablePerformanceLogRecords mutablePerformanceLogRecords;
private final boolean aclLogging;
private final MutableAclLogRecords mutableAclLogRecords;
/**
* Creates a LoggingContextAwareCallable that wraps the given {@link Callable}.
*
* @param callable Callable that should be wrapped.
* @param mutablePerformanceLogRecords instance of {@link MutablePerformanceLogRecords} to which
* performance log records that are created from the runnable are added
* @param mutableAclLogRecords instance of {@link MutableAclLogRecords} to which ACL log records
* that are created from the runnable are added
*/
LoggingContextAwareCallable(
Callable<T> callable,
MutablePerformanceLogRecords mutablePerformanceLogRecords,
MutableAclLogRecords mutableAclLogRecords) {
this.callable = callable;
this.callingThread = Thread.currentThread();
this.tags = LoggingContext.getInstance().getTagsAsMap();
this.forceLogging = LoggingContext.getInstance().isLoggingForced();
this.performanceLogging = LoggingContext.getInstance().isPerformanceLogging();
this.mutablePerformanceLogRecords = mutablePerformanceLogRecords;
this.aclLogging = LoggingContext.getInstance().isAclLogging();
this.mutableAclLogRecords = mutableAclLogRecords;
}
@Override
public T call() throws Exception {
if (callingThread.equals(Thread.currentThread())) {
// propagation of logging context is not needed
return callable.call();
}
LoggingContext loggingCtx = LoggingContext.getInstance();
if (!loggingCtx.isEmpty()) {
logger.atWarning().log("Logging context is not empty: %s", loggingCtx);
}
// propagate logging context
loggingCtx.setTags(tags);
loggingCtx.forceLogging(forceLogging);
loggingCtx.performanceLogging(performanceLogging);
loggingCtx.setMutablePerformanceLogRecords(mutablePerformanceLogRecords);
loggingCtx.aclLogging(aclLogging);
loggingCtx.setMutableAclLogRecords(mutableAclLogRecords);
try {
return callable.call();
} finally {
// Cleanup logging context. This is important if the thread is pooled and reused.
loggingCtx.clear();
}
}
}