org.glowroot.agent.impl.TraceCollector Maven / Gradle / Ivy
/*
* Copyright 2011-2019 the original author or authors.
*
* 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 org.glowroot.agent.impl;
import java.util.Collection;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.concurrent.BlockingQueue;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.atomic.AtomicInteger;
import org.glowroot.agent.shaded.org.glowroot.agent.it.harness.shaded.com.google.common.base.Ticker;
import org.glowroot.agent.shaded.org.glowroot.agent.it.harness.shaded.com.google.common.collect.ImmutableMap;
import org.glowroot.agent.shaded.org.glowroot.agent.it.harness.shaded.com.google.common.collect.Lists;
import org.glowroot.agent.shaded.org.glowroot.agent.it.harness.shaded.com.google.common.collect.Maps;
import org.glowroot.agent.shaded.org.glowroot.agent.it.harness.shaded.com.google.common.collect.Queues;
import org.checkerframework.checker.nullness.qual.Nullable;
import org.immutables.value.Value;
import org.glowroot.agent.shaded.org.glowroot.agent.shaded.org.slf4j.Logger;
import org.glowroot.agent.shaded.org.glowroot.agent.shaded.org.slf4j.LoggerFactory;
import org.glowroot.agent.collector.Collector;
import org.glowroot.agent.collector.Collector.TraceReader;
import org.glowroot.agent.config.ConfigService;
import org.glowroot.agent.plugin.api.config.ConfigListener;
import org.glowroot.agent.util.RateLimitedLogger;
import org.glowroot.agent.util.ThreadFactories;
import org.glowroot.agent.shaded.org.glowroot.common.config.TransactionConfig;
import org.glowroot.agent.shaded.org.glowroot.common.config.TransactionConfig.SlowThresholdOverride;
import org.glowroot.agent.shaded.org.glowroot.common.util.Clock;
import org.glowroot.agent.shaded.org.glowroot.common.util.OnlyUsedByTests;
import static java.util.concurrent.TimeUnit.MILLISECONDS;
import static java.util.concurrent.TimeUnit.SECONDS;
public class TraceCollector {
private static final Logger logger = LoggerFactory.getLogger(TraceCollector.class);
// back pressure on writing captured data to disk/network
private static final int PENDING_LIMIT = 50;
private final ExecutorService dedicatedExecutor;
private final Collector collector;
private final Clock clock;
private final Ticker ticker;
// covers normal complete, partial complete and partial incomplete separately
private final BlockingQueue pendingTraces =
Queues.newLinkedBlockingQueue(PENDING_LIMIT * 3);
private final AtomicInteger normalCompletePendingCount = new AtomicInteger();
private final AtomicInteger partialCompletePendingCount = new AtomicInteger();
private final AtomicInteger partialIncompletePendingCount = new AtomicInteger();
private final RateLimitedLogger backPressureLogger =
new RateLimitedLogger(TraceCollector.class);
// visibility is provided by memoryBarrier in org.glowroot.config.ConfigService
private Map slowThresholdOverrides = ImmutableMap.of();
// visibility is provided by memoryBarrier in org.glowroot.config.ConfigService
private long defaultSlowThresholdNanos;
private volatile boolean closed;
public TraceCollector(final ConfigService configService, Collector collector, Clock clock,
Ticker ticker) {
this.collector = collector;
this.clock = clock;
this.ticker = ticker;
dedicatedExecutor = Executors
.newSingleThreadExecutor(ThreadFactories.create("Glowroot-Trace-Collector"));
dedicatedExecutor.execute(new TraceCollectorLoop());
configService.addConfigListener(new UpdateLocalConfig(configService));
}
public boolean shouldStoreSlow(Transaction transaction) {
if (transaction.isPartiallyStored()) {
return true;
}
long durationNanos = transaction.getDurationNanos();
// check if trace-specific store threshold was set
long slowThresholdMillis = transaction.getSlowThresholdMillisOverride();
if (slowThresholdMillis != Transaction.USE_GENERAL_STORE_THRESHOLD) {
return durationNanos >= MILLISECONDS.toNanos(slowThresholdMillis);
}
// check if there is a matching transaction type / transaction name / user specific slow
// threshold
if (!slowThresholdOverrides.isEmpty()) {
SlowThresholdOverridesForType slowThresholdOverrideForType =
slowThresholdOverrides.get(transaction.getTransactionType());
if (slowThresholdOverrideForType != null) {
String transactionName = transaction.getTransactionName();
String user = transaction.getUser();
// user specific thresholds take precedence
Map userThresholds =
slowThresholdOverrideForType.userThresholds();
if (!userThresholds.isEmpty() && !user.isEmpty()) {
String userLowerCase = user.toLowerCase(Locale.ENGLISH);
SlowThresholdOverridesForUser slowThresholdOverridesForUser =
userThresholds.get(userLowerCase);
if (slowThresholdOverridesForUser != null) {
Long slowThresholdNanos =
getSlowThreshold(slowThresholdOverridesForUser,
transactionName);
if (slowThresholdNanos != null) {
return durationNanos >= slowThresholdNanos;
}
}
}
Long slowThresholdNanos =
getSlowThreshold(slowThresholdOverrideForType, transactionName);
if (slowThresholdNanos != null) {
return durationNanos >= slowThresholdNanos;
}
slowThresholdNanos = slowThresholdOverrideForType.defaultThresholdNanos();
if (slowThresholdNanos != null) {
return durationNanos >= slowThresholdNanos;
}
}
}
// fall back to default slow trace threshold
return durationNanos >= defaultSlowThresholdNanos;
}
public boolean shouldStoreError(Transaction transaction) {
return transaction.getErrorMessage() != null;
}
public Collection getPendingTransactions() {
List pendingTransactions = Lists.newArrayList();
for (PendingTrace pendingTrace : pendingTraces) {
pendingTransactions.add(pendingTrace.transaction());
}
return pendingTransactions;
}
@OnlyUsedByTests
public void close() throws InterruptedException {
closed = true;
// shutdownNow() is needed here to send interrupt to collector thread
dedicatedExecutor.shutdownNow();
if (!dedicatedExecutor.awaitTermination(10, SECONDS)) {
throw new IllegalStateException("Could not terminate executor");
}
}
void collectTrace(Transaction transaction) {
boolean slow = shouldStoreSlow(transaction);
if (!slow && !shouldStoreError(transaction)) {
return;
}
// don't need to worry about race condition since only ever called from a single thread
if (transaction.isPartiallyStored()
&& partialCompletePendingCount.get() >= PENDING_LIMIT) {
backPressureLogger.warn("not storing a completed (and once partial) trace because of an"
+ " excessive backlog of {} completed (and once partial) traces already waiting"
+ " to be stored", PENDING_LIMIT);
return;
} else if (!transaction.isPartiallyStored()
&& normalCompletePendingCount.get() >= PENDING_LIMIT) {
backPressureLogger.warn("not storing a completed trace because of an excessive backlog"
+ " of {} completed traces already waiting to be stored", PENDING_LIMIT);
return;
}
PendingTrace pendingTransaction = ImmutablePendingTrace.builder()
.transaction(transaction)
.slow(slow)
.partial(false)
.build();
if (!pendingTraces.offer(pendingTransaction)) {
// this should never happen
backPressureLogger.warn("not storing a trace because of an excessive backlog of {}"
+ " traces already waiting to be stored", PENDING_LIMIT * 3);
}
}
public void storePartialTrace(Transaction transaction) {
// don't need to worry about race condition since only ever called from a single thread
if (partialIncompletePendingCount.get() >= PENDING_LIMIT) {
backPressureLogger.warn("not storing a partial trace because of an excessive backlog of"
+ " {} partial traces already waiting to be stored", PENDING_LIMIT);
return;
}
PendingTrace pendingTransaction = ImmutablePendingTrace.builder()
.transaction(transaction)
.slow(false)
.partial(true)
.build();
if (!pendingTraces.offer(pendingTransaction)) {
// this should never happen
backPressureLogger.warn("not storing a trace because of an excessive backlog of {}"
+ " traces already waiting to be stored", PENDING_LIMIT * 3);
}
}
private static @Nullable Long getSlowThreshold(
SlowThresholdOverridesForType slowThresholdOverridesForType, String transactionName) {
Long slowThreshold = slowThresholdOverridesForType.thresholdNanos().get(transactionName);
return slowThreshold == null ? slowThresholdOverridesForType.defaultThresholdNanos()
: slowThreshold;
}
private static @Nullable Long getSlowThreshold(
SlowThresholdOverridesForUser slowThresholdOverridesForUser, String transactionName) {
Long slowThreshold = slowThresholdOverridesForUser.thresholdNanos().get(transactionName);
return slowThreshold == null ? slowThresholdOverridesForUser.defaultThresholdNanos()
: slowThreshold;
}
private class UpdateLocalConfig implements ConfigListener {
private final ConfigService configService;
private UpdateLocalConfig(ConfigService configService) {
this.configService = configService;
}
@Override
public void onChange() {
TransactionConfig transactionConfig = configService.getTransactionConfig();
Map slowThresholdOverrides =
Maps.newHashMap();
for (SlowThresholdOverride slowThresholdOverride : transactionConfig
.slowThresholdOverrides()) {
String transactionType = slowThresholdOverride.transactionType();
SlowThresholdOverridesForTypeBuilder slowThresholdOverrideForType =
slowThresholdOverrides.get(transactionType);
if (slowThresholdOverrideForType == null) {
slowThresholdOverrideForType = new SlowThresholdOverridesForTypeBuilder();
slowThresholdOverrides.put(transactionType, slowThresholdOverrideForType);
}
String transactionName = slowThresholdOverride.transactionName();
String user = slowThresholdOverride.user();
long thresholdNanos = MILLISECONDS.toNanos(slowThresholdOverride.thresholdMillis());
if (user.isEmpty()) {
if (transactionName.isEmpty()) {
slowThresholdOverrideForType.defaultThresholdNanos = thresholdNanos;
} else {
slowThresholdOverrideForType.thresholdNanos.put(transactionName,
thresholdNanos);
}
} else {
String userLowerCase = user.toLowerCase(Locale.ENGLISH);
SlowThresholdOverridesForUserBuilder slowThresholdOverridesForUserBuilder =
slowThresholdOverrideForType.userThresholdNanos.get(userLowerCase);
if (slowThresholdOverridesForUserBuilder == null) {
slowThresholdOverridesForUserBuilder =
new SlowThresholdOverridesForUserBuilder();
slowThresholdOverrideForType.userThresholdNanos.put(userLowerCase,
slowThresholdOverridesForUserBuilder);
}
if (transactionName.isEmpty()) {
slowThresholdOverridesForUserBuilder.defaultThresholdNanos =
thresholdNanos;
} else {
slowThresholdOverridesForUserBuilder.thresholdNanos.put(transactionName,
thresholdNanos);
}
}
}
Map builder = Maps.newHashMap();
for (Map.Entry entry : slowThresholdOverrides
.entrySet()) {
builder.put(entry.getKey(), entry.getValue().toImmutable());
}
TraceCollector.this.slowThresholdOverrides = ImmutableMap.copyOf(builder);
defaultSlowThresholdNanos =
MILLISECONDS.toNanos(transactionConfig.slowThresholdMillis());
}
}
private class TraceCollectorLoop implements Runnable {
@Override
public void run() {
while (!closed) {
try {
PendingTrace pendingTrace = pendingTraces.take();
if (pendingTrace.partial()) {
collectPartial(pendingTrace.transaction());
} else {
collectCompleted(pendingTrace.transaction(), pendingTrace.slow());
}
} catch (InterruptedException e) {
// probably shutdown requested (see close method above)
logger.debug(e.getMessage(), e);
} catch (Throwable e) {
// log and continue processing
logger.error(e.getMessage(), e);
}
}
}
private void collectPartial(Transaction transaction) throws Exception {
TraceReader traceReader = TraceCreator.createTraceReaderForPartial(transaction,
clock.currentTimeMillis(), ticker.read());
// one last check if transaction has completed
if (!transaction.isCompleted()) {
transaction.setPartiallyStored();
collector.collectTrace(traceReader);
}
}
private void collectCompleted(Transaction transaction, boolean slow) throws Exception {
TraceReader traceReader = TraceCreator.createTraceReaderForCompleted(
transaction, slow);
collector.collectTrace(traceReader);
}
}
@Value.Immutable
public interface PendingTrace {
Transaction transaction();
boolean slow();
boolean partial();
}
@Value.Immutable
interface SlowThresholdOverridesForType {
@Nullable
Long defaultThresholdNanos();
Map thresholdNanos();
Map userThresholds(); // key is user
}
@Value.Immutable
interface SlowThresholdOverridesForUser {
@Nullable
Long defaultThresholdNanos();
Map thresholdNanos();
}
private static class SlowThresholdOverridesForTypeBuilder {
private @Nullable Long defaultThresholdNanos;
private Map userThresholdNanos =
Maps.newHashMap();
private Map thresholdNanos = Maps.newHashMap();
private SlowThresholdOverridesForType toImmutable() {
ImmutableSlowThresholdOverridesForType.Builder builder =
ImmutableSlowThresholdOverridesForType.builder()
.defaultThresholdNanos(defaultThresholdNanos);
for (Map.Entry entry : userThresholdNanos
.entrySet()) {
builder.putUserThresholds(entry.getKey(), entry.getValue().toImmutable());
}
return builder.putAllThresholdNanos(thresholdNanos)
.build();
}
}
private static class SlowThresholdOverridesForUserBuilder {
private @Nullable Long defaultThresholdNanos;
private Map thresholdNanos = Maps.newHashMap();
private SlowThresholdOverridesForUser toImmutable() {
return ImmutableSlowThresholdOverridesForUser.builder()
.defaultThresholdNanos(defaultThresholdNanos)
.putAllThresholdNanos(thresholdNanos)
.build();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy