Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
org.glowroot.central.CollectorServiceImpl Maven / Gradle / Ivy
/*
* Copyright 2017-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.central;
import java.text.SimpleDateFormat;
import java.util.ArrayList;
import java.util.Date;
import java.util.HashMap;
import java.util.List;
import java.util.ListIterator;
import java.util.Map;
import java.util.concurrent.Semaphore;
import java.util.concurrent.atomic.AtomicInteger;
import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;
import io.grpc.Status;
import io.grpc.stub.StreamObserver;
import org.checkerframework.checker.nullness.qual.MonotonicNonNull;
import org.checkerframework.checker.nullness.qual.RequiresNonNull;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.glowroot.agent.api.Instrumentation;
import org.glowroot.central.repo.ActiveAgentDao;
import org.glowroot.central.repo.AgentConfigDao;
import org.glowroot.central.repo.AgentDisplayDao;
import org.glowroot.central.repo.AggregateDao;
import org.glowroot.central.repo.EnvironmentDao;
import org.glowroot.central.repo.GaugeValueDao;
import org.glowroot.central.repo.HeartbeatDao;
import org.glowroot.central.repo.SchemaUpgrade;
import org.glowroot.central.repo.TraceDao;
import org.glowroot.central.repo.V09AgentRollupDao;
import org.glowroot.central.util.MoreFutures;
import org.glowroot.common.util.Clock;
import org.glowroot.wire.api.model.AgentConfigOuterClass.AgentConfig;
import org.glowroot.wire.api.model.AggregateOuterClass.Aggregate;
import org.glowroot.wire.api.model.AggregateOuterClass.OldAggregatesByType;
import org.glowroot.wire.api.model.AggregateOuterClass.OldTransactionAggregate;
import org.glowroot.wire.api.model.CollectorServiceGrpc;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.AggregateResponseMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.AggregateStreamMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.AggregateStreamMessage.AggregateStreamHeader;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.AggregateStreamMessage.OverallAggregate;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.AggregateStreamMessage.TransactionAggregate;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.EmptyMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.GaugeValueMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.GaugeValueMessage.GaugeValue;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.GaugeValueResponseMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.InitMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.InitResponse;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.LogMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.LogMessage.LogEvent;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.LogMessage.LogEvent.Level;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.OldAggregateMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.OldTraceMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.TraceStreamMessage;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.TraceStreamMessage.TraceStreamCounts;
import org.glowroot.wire.api.model.CollectorServiceOuterClass.TraceStreamMessage.TraceStreamHeader;
import org.glowroot.wire.api.model.ProfileOuterClass.Profile;
import org.glowroot.wire.api.model.Proto;
import org.glowroot.wire.api.model.TraceOuterClass.Trace;
import static com.google.common.base.Preconditions.checkNotNull;
import static com.google.common.base.Preconditions.checkState;
import static java.util.concurrent.TimeUnit.HOURS;
import static java.util.concurrent.TimeUnit.MINUTES;
class CollectorServiceImpl extends CollectorServiceGrpc.CollectorServiceImplBase {
private static final Logger logger = LoggerFactory.getLogger(CollectorServiceImpl.class);
private final AgentDisplayDao agentDisplayDao;
private final AgentConfigDao agentConfigDao;
private final ActiveAgentDao activeAgentDao;
private final EnvironmentDao environmentDao;
private final HeartbeatDao heartbeatDao;
private final AggregateDao aggregateDao;
private final GaugeValueDao gaugeValueDao;
private final TraceDao traceDao;
private final V09AgentRollupDao v09AgentRollupDao;
private final GrpcCommon grpcCommon;
private final CentralAlertingService centralAlertingService;
private final Clock clock;
private final String version;
private volatile long currentMinute;
private final AtomicInteger nextDelay = new AtomicInteger();
private final LoadingCache throttlePerAgentId = CacheBuilder.newBuilder()
.weakValues()
.build(new CacheLoader() {
@Override
public Semaphore load(String key) throws Exception {
return new Semaphore(1);
}
});
CollectorServiceImpl(AgentDisplayDao agentDisplayDao, AgentConfigDao agentConfigDao,
ActiveAgentDao activeAgentDao, EnvironmentDao environmentDao, HeartbeatDao heartbeatDao,
AggregateDao aggregateDao, GaugeValueDao gaugeValueDao, TraceDao traceDao,
V09AgentRollupDao v09AgentRollupDao, GrpcCommon grpcCommon,
CentralAlertingService centralAlertingService, Clock clock, String version) {
this.agentDisplayDao = agentDisplayDao;
this.agentConfigDao = agentConfigDao;
this.activeAgentDao = activeAgentDao;
this.environmentDao = environmentDao;
this.heartbeatDao = heartbeatDao;
this.aggregateDao = aggregateDao;
this.gaugeValueDao = gaugeValueDao;
this.traceDao = traceDao;
this.v09AgentRollupDao = v09AgentRollupDao;
this.grpcCommon = grpcCommon;
this.centralAlertingService = centralAlertingService;
this.clock = clock;
this.version = version;
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Init",
traceHeadline = "Collect init: {{0.agentId}}", timer = "init")
@Override
public void collectInit(InitMessage request, StreamObserver responseObserver) {
String agentId = request.getAgentId();
String v09AgentRollupId = request.getV09AgentRollupId();
if (!v09AgentRollupId.isEmpty()) {
// handle agents prior to 0.10.0
String v09AgentId = agentId;
agentId = GrpcCommon.convertFromV09AgentRollupId(v09AgentRollupId) + v09AgentId;
try {
v09AgentRollupDao.store(v09AgentId, v09AgentRollupId);
} catch (Throwable t) {
logger.error("{} - {}", agentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
}
AgentConfig updatedAgentConfig;
try {
updatedAgentConfig = SchemaUpgrade.upgradeOldAgentConfig(request.getAgentConfig());
updatedAgentConfig = agentConfigDao.store(agentId, updatedAgentConfig,
request.getOverwriteExistingAgentConfig());
environmentDao.store(agentId, request.getEnvironment());
MoreFutures.waitForAll(activeAgentDao.insert(agentId, clock.currentTimeMillis()));
} catch (Throwable t) {
logger.error("{} - {}", agentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
logger.info("agent connected: {}, version {}", agentId,
request.getEnvironment().getJavaInfo().getGlowrootAgentVersion());
InitResponse.Builder response = InitResponse.newBuilder()
.setGlowrootCentralVersion(version);
if (!updatedAgentConfig.equals(request.getAgentConfig())) {
response.setAgentConfig(updatedAgentConfig);
}
responseObserver.onNext(response.build());
responseObserver.onCompleted();
}
@Override
public StreamObserver collectAggregateStream(
StreamObserver responseObserver) {
return new AggregateStreamObserver(responseObserver);
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Aggregates",
traceHeadline = "Collect aggregates: {{0.agentId}}", timer = "aggregates")
@Override
public void collectAggregates(OldAggregateMessage request,
StreamObserver responseObserver) {
List sharedQueryTexts;
List oldSharedQueryTexts = request.getOldSharedQueryTextList();
if (oldSharedQueryTexts.isEmpty()) {
sharedQueryTexts = request.getSharedQueryTextList();
} else {
// handle agents prior to 0.9.3
sharedQueryTexts = new ArrayList<>();
for (String oldSharedQueryText : oldSharedQueryTexts) {
sharedQueryTexts.add(Aggregate.SharedQueryText.newBuilder()
.setFullText(oldSharedQueryText)
.build());
}
}
throttleCollectAggregates(request.getAgentId(), false,
getFutureProofAggregateCaptureTime(request.getCaptureTime()),
sharedQueryTexts, request.getAggregatesByTypeList(), responseObserver);
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Gauges",
traceHeadline = "Collect gauge values: {{0.agentId}}", timer = "gauges")
@Override
public void collectGaugeValues(GaugeValueMessage request,
StreamObserver responseObserver) {
throttledCollectGaugeValues(request, responseObserver);
}
@Override
public StreamObserver collectTraceStream(
StreamObserver responseObserver) {
return new TraceStreamObserver(responseObserver);
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Trace",
traceHeadline = "Collect trace: {{0.agentId}}", timer = "trace")
@Override
public void collectTrace(OldTraceMessage request,
StreamObserver responseObserver) {
throttledCollectTrace(request.getAgentId(), false, request.getTrace(), responseObserver);
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Log",
traceHeadline = "Log: {{0.agentId}}", timer = "log")
@Override
public void log(LogMessage request, StreamObserver responseObserver) {
String agentId;
try {
agentId = grpcCommon.getAgentId(request.getAgentId(), request.getPostV09());
} catch (Throwable t) {
logger.error("{} - {}",
getAgentIdForLogging(request.getAgentId(), request.getPostV09()),
t.getMessage(), t);
responseObserver.onError(t);
return;
}
try {
LogEvent logEvent = request.getLogEvent();
Level level = logEvent.getLevel();
String formattedTimestamp = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss.SSS")
.format(new Date(logEvent.getTimestamp()));
if (logEvent.hasThrowable()) {
log(level, "{} - {} {} {} - {}\n{}", agentId, formattedTimestamp,
level, logEvent.getLoggerName(), logEvent.getMessage(),
toString(logEvent.getThrowable()));
} else {
log(level, "{} - {} {} {} - {}", agentId, formattedTimestamp, level,
logEvent.getLoggerName(), logEvent.getMessage());
}
} catch (Throwable t) {
logger.error("{} - {}", agentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
responseObserver.onNext(EmptyMessage.getDefaultInstance());
responseObserver.onCompleted();
}
private void throttleCollectAggregates(String agentId, boolean postV09, long captureTime,
List sharedQueryTexts,
List aggregatesByTypeList,
StreamObserver responseObserver) {
throttle(agentId, postV09, "aggregate", responseObserver, new Runnable() {
@Override
public void run() {
collectAggregatesUnderThrottle(agentId, postV09, captureTime, sharedQueryTexts,
aggregatesByTypeList, responseObserver);
}
});
}
private void throttledCollectGaugeValues(GaugeValueMessage request,
StreamObserver responseObserver) {
throttle(request.getAgentId(), request.getPostV09(), "gauge value", responseObserver,
new Runnable() {
@Override
public void run() {
collectGaugeValuesUnderThrottle(request, responseObserver);
}
});
}
private void throttledCollectTrace(String agentId, boolean postV09, Trace trace,
StreamObserver responseObserver) {
throttle(agentId, postV09, "trace", responseObserver, new Runnable() {
@Override
public void run() {
collectTraceUnderThrottle(agentId, postV09, trace, responseObserver);
}
});
}
private void throttle(String agentId, boolean postV09, String collectionType,
StreamObserver responseObserver, Runnable runnable) {
Semaphore semaphore = throttlePerAgentId.getUnchecked(agentId);
boolean acquired;
try {
acquired = semaphore.tryAcquire(1, MINUTES);
} catch (InterruptedException e) {
// probably shutdown requested
responseObserver.onError(e);
return;
}
if (!acquired) {
logger.warn("{} - {} collection rejected due to backlog",
getAgentIdForLogging(agentId, postV09), collectionType);
responseObserver.onError(Status.RESOURCE_EXHAUSTED
.withDescription("collection rejected due to backlog")
.asRuntimeException());
return;
}
try {
runnable.run();
} finally {
semaphore.release();
}
}
private void collectAggregatesUnderThrottle(String agentId, boolean postV09, long captureTime,
List sharedQueryTexts,
List aggregatesByTypeList,
StreamObserver responseObserver) {
String postV09AgentId;
try {
postV09AgentId = grpcCommon.getAgentId(agentId, postV09);
} catch (Throwable t) {
logger.error("{} - {}", getAgentIdForLogging(agentId, postV09), t.getMessage(), t);
responseObserver.onError(t);
return;
}
try {
aggregateDao.store(postV09AgentId, captureTime, aggregatesByTypeList, sharedQueryTexts);
} catch (Throwable t) {
logger.error("{} - {}", postV09AgentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
String agentDisplay;
try {
agentDisplay = agentDisplayDao.readFullDisplay(postV09AgentId);
} catch (Exception e) {
logger.error("{} - {}", postV09AgentId, e.getMessage(), e);
responseObserver.onError(e);
return;
}
try {
centralAlertingService.checkForDeletedAlerts(postV09AgentId);
centralAlertingService.checkAggregateAlertsAsync(postV09AgentId, agentDisplay,
captureTime);
} catch (InterruptedException e) {
// probably shutdown requested
logger.debug(e.getMessage(), e);
}
responseObserver.onNext(AggregateResponseMessage.newBuilder()
.setNextDelayMillis(getNextDelayMillis())
.build());
responseObserver.onCompleted();
}
private void collectGaugeValuesUnderThrottle(GaugeValueMessage request,
StreamObserver responseObserver) {
String postV09AgentId;
try {
postV09AgentId = grpcCommon.getAgentId(request.getAgentId(), request.getPostV09());
} catch (Throwable t) {
logger.error("{} - {}",
getAgentIdForLogging(request.getAgentId(), request.getPostV09()),
t.getMessage(), t);
responseObserver.onError(t);
return;
}
long maxCaptureTime = 0;
try {
List gaugeValues = getFutureProofGaugeValues(request.getGaugeValueList());
gaugeValueDao.store(postV09AgentId, gaugeValues);
for (GaugeValue gaugeValue : gaugeValues) {
maxCaptureTime = Math.max(maxCaptureTime, gaugeValue.getCaptureTime());
}
} catch (Throwable t) {
logger.error("{} - {}", postV09AgentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
try {
heartbeatDao.store(postV09AgentId);
} catch (Throwable t) {
logger.error("{} - {}", postV09AgentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
String agentDisplay;
try {
agentDisplay = agentDisplayDao.readFullDisplay(postV09AgentId);
} catch (Throwable t) {
logger.error("{} - {}", postV09AgentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
try {
centralAlertingService.checkForDeletedAlerts(postV09AgentId);
centralAlertingService.checkGaugeAndHeartbeatAlertsAsync(postV09AgentId, agentDisplay,
maxCaptureTime);
} catch (InterruptedException e) {
// probably shutdown requested
logger.debug(e.getMessage(), e);
}
boolean resendInit;
try {
resendInit = agentConfigDao.read(postV09AgentId) == null
|| environmentDao.read(postV09AgentId) == null;
} catch (Throwable t) {
// log as error, but not worth failing for this
logger.error("{} - {}", postV09AgentId, t.getMessage(), t);
resendInit = false;
}
responseObserver.onNext(GaugeValueResponseMessage.newBuilder()
.setResendInit(resendInit)
.build());
responseObserver.onCompleted();
}
private void collectTraceUnderThrottle(String agentId, boolean postV09, Trace trace,
StreamObserver responseObserver) {
String postV09AgentId;
try {
postV09AgentId = grpcCommon.getAgentId(agentId, postV09);
} catch (Throwable t) {
logger.error("{} - {}", getAgentIdForLogging(agentId, postV09), t.getMessage(), t);
responseObserver.onError(t);
return;
}
try {
traceDao.store(postV09AgentId, getFutureProofTrace(trace));
} catch (Throwable t) {
logger.error("{} - {}", postV09AgentId, t.getMessage(), t);
responseObserver.onError(t);
return;
}
responseObserver.onNext(EmptyMessage.getDefaultInstance());
responseObserver.onCompleted();
}
private long getFutureProofAggregateCaptureTime(long captureTime) {
long currentTimeMillis = clock.currentTimeMillis();
if (tooFarInTheFuture(captureTime, currentTimeMillis)) {
// too far in the future just use current time
return (long) Math.floor(currentTimeMillis / 60000.0) * 60000;
} else {
return captureTime;
}
}
private List getFutureProofGaugeValues(List gaugeValues) {
List futureProofGaugeValues = new ArrayList<>(gaugeValues);
long currentTimeMillis = clock.currentTimeMillis();
for (ListIterator i = futureProofGaugeValues.listIterator(); i.hasNext();) {
GaugeValue gaugeValue = i.next();
if (tooFarInTheFuture(gaugeValue.getCaptureTime(), currentTimeMillis)) {
// too far in the future just use current time
i.set(gaugeValue.toBuilder()
.setCaptureTime(currentTimeMillis)
.build());
}
}
return futureProofGaugeValues;
}
private Trace getFutureProofTrace(Trace trace) {
long currentTimeMillis = clock.currentTimeMillis();
if (tooFarInTheFuture(trace.getHeader().getCaptureTime(), currentTimeMillis)) {
// too far in the future just use current time
return trace.toBuilder()
.setHeader(trace.getHeader().toBuilder()
.setCaptureTime(currentTimeMillis))
.build();
} else {
return trace;
}
}
private int getNextDelayMillis() {
long currentishTimeMillis = clock.currentTimeMillis() + 10000;
if (currentishTimeMillis > currentMinute) {
// race condition here is ok, at worst results in resetting nextDelay multiple times
nextDelay.set(0);
currentMinute = (long) Math.ceil(currentishTimeMillis / 60000.0) * 60000;
}
// spread out aggregate collections 100 milliseconds a part, rolling over at 10 seconds
return nextDelay.getAndAdd(100) % 10000;
}
private String getAgentIdForLogging(String agentId, boolean postV09) {
return grpcCommon.getAgentIdForLogging(agentId, postV09);
}
// far in the future capture times cause major problems for Cassandra TWCS since then every
// rollup of recent data must consider an (increasingly) old sstable and (worse) all of the
// sstables in between (because of possibility of tombstones in more recent sstables
// invalidating the data read from the old sstable)
//
// another problem with far in the future capture times is that the adjusted TTL (see
// Common.getAdjustedTTL()) will be really large, preventing that data from expiring along with
// its friends (probably also contributing to TWCS problems)
private static boolean tooFarInTheFuture(long captureTime, long currentTimeMillis) {
return captureTime - currentTimeMillis > HOURS.toMillis(1);
}
private static String toString(Proto.Throwable t) {
StringBuilder sb = new StringBuilder();
sb.append(t.getClassName());
String message = t.getMessage();
if (!message.isEmpty()) {
sb.append(": ");
sb.append(message);
}
for (Proto.StackTraceElement stackTraceElement : t.getStackTraceElementList()) {
sb.append("\n\tat ");
sb.append(new StackTraceElement(stackTraceElement.getClassName(),
stackTraceElement.getMethodName(), stackTraceElement.getFileName(),
stackTraceElement.getLineNumber()));
}
int framesInCommonWithEnclosing = t.getFramesInCommonWithEnclosing();
if (framesInCommonWithEnclosing > 0) {
sb.append("\n\t... ");
sb.append(framesInCommonWithEnclosing);
sb.append(" more");
}
if (t.hasCause()) {
sb.append("\nCaused by: ");
sb.append(toString(t.getCause()));
}
return sb.toString();
}
private static void log(Level level, String format, Object... arguments) {
switch (level) {
case ERROR:
logger.error(format, arguments);
break;
case WARN:
logger.warn(format, arguments);
break;
default:
logger.info(format, arguments);
break;
}
}
private final class AggregateStreamObserver implements StreamObserver {
private final StreamObserver responseObserver;
private @MonotonicNonNull AggregateStreamHeader streamHeader;
private List sharedQueryTexts = new ArrayList<>();
private Map aggregatesByTypeMap = new HashMap<>();
private AggregateStreamObserver(StreamObserver responseObserver) {
this.responseObserver = responseObserver;
}
@Override
public void onNext(AggregateStreamMessage value) {
try {
onNextInternal(value);
} catch (Throwable t) {
logError(t);
throw t;
}
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Aggregates",
traceHeadline = "Collect aggregates: {{this.streamHeader.agentId}}",
timer = "aggregates")
@Override
public void onCompleted() {
try {
onCompletedInternal();
} catch (Throwable t) {
logError(t);
throw t;
}
}
@Override
public void onError(Throwable t) {
logError(t);
}
private void onNextInternal(AggregateStreamMessage value) {
switch (value.getMessageCase()) {
case STREAM_HEADER:
streamHeader = value.getStreamHeader();
break;
case SHARED_QUERY_TEXT:
sharedQueryTexts.add(value.getSharedQueryText());
break;
case OVERALL_AGGREGATE:
OverallAggregate overallAggregate = value.getOverallAggregate();
String transactionType = overallAggregate.getTransactionType();
aggregatesByTypeMap.put(transactionType, OldAggregatesByType.newBuilder()
.setTransactionType(transactionType)
.setOverallAggregate(overallAggregate.getAggregate()));
break;
case TRANSACTION_AGGREGATE:
TransactionAggregate transactionAggregate = value.getTransactionAggregate();
OldAggregatesByType.Builder builder = checkNotNull(
aggregatesByTypeMap.get(transactionAggregate.getTransactionType()));
builder.addTransactionAggregate(OldTransactionAggregate.newBuilder()
.setTransactionName(transactionAggregate.getTransactionName())
.setAggregate(transactionAggregate.getAggregate())
.build());
break;
default:
throw new RuntimeException("Unexpected message: " + value.getMessageCase());
}
}
private void onCompletedInternal() {
checkNotNull(streamHeader);
List aggregatesByTypeList = new ArrayList<>();
for (OldAggregatesByType.Builder aggregatesByType : aggregatesByTypeMap.values()) {
aggregatesByTypeList.add(aggregatesByType.build());
}
throttleCollectAggregates(streamHeader.getAgentId(), streamHeader.getPostV09(),
getFutureProofAggregateCaptureTime(streamHeader.getCaptureTime()),
sharedQueryTexts,
aggregatesByTypeList, responseObserver);
}
private void logError(Throwable t) {
if (streamHeader == null) {
logger.error(t.getMessage(), t);
} else {
logger.error("{} - {}", grpcCommon.getAgentIdForLogging(streamHeader.getAgentId(),
streamHeader.getPostV09()), t.getMessage(), t);
}
}
}
private final class TraceStreamObserver implements StreamObserver {
private final StreamObserver responseObserver;
private @MonotonicNonNull TraceStreamHeader streamHeader;
private List sharedQueryTexts = new ArrayList<>();
private @MonotonicNonNull Trace trace;
private List entries = new ArrayList<>();
private List queries = new ArrayList<>();
private @MonotonicNonNull Profile mainThreadProfile;
private @MonotonicNonNull Profile auxThreadProfile;
private Trace. /*@MonotonicNonNull*/ Header header;
private @MonotonicNonNull TraceStreamCounts streamCounts;
private TraceStreamObserver(StreamObserver responseObserver) {
this.responseObserver = responseObserver;
}
@Override
public void onNext(TraceStreamMessage value) {
try {
onNextInternal(value);
} catch (Throwable t) {
logError(t);
throw t;
}
}
@Instrumentation.Transaction(transactionType = "gRPC", transactionName = "Trace",
traceHeadline = "Collect trace: {{this.streamHeader.agentId}}", timer = "trace")
@Override
public void onCompleted() {
try {
onCompletedInternal();
} catch (Throwable t) {
logError(t);
throw t;
}
}
@Override
public void onError(Throwable t) {
logError(t);
}
private void onNextInternal(TraceStreamMessage value) {
switch (value.getMessageCase()) {
case STREAM_HEADER:
streamHeader = value.getStreamHeader();
break;
case SHARED_QUERY_TEXT:
sharedQueryTexts.add(value.getSharedQueryText());
break;
case TRACE:
// this is for 0.9.12 and prior agents
trace = value.getTrace();
break;
case ENTRY:
entries.add(value.getEntry());
break;
case QUERIES:
queries.addAll(value.getQueries().getQueryList());
break;
case MAIN_THREAD_PROFILE:
mainThreadProfile = value.getMainThreadProfile();
break;
case AUX_THREAD_PROFILE:
auxThreadProfile = value.getAuxThreadProfile();
break;
case HEADER:
header = value.getHeader();
break;
case STREAM_COUNTS:
streamCounts = value.getStreamCounts();
break;
default:
throw new RuntimeException("Unexpected message: " + value.getMessageCase());
}
}
private void onCompletedInternal() {
checkNotNull(streamHeader);
if (trace == null) {
// this is for 0.9.13 and later agents
checkNotNull(streamCounts);
if (!isEverythingReceived()) {
// no point in calling onError to force re-try since gRPC maxMessageSize limit
// will just be hit again
responseObserver.onNext(EmptyMessage.getDefaultInstance());
responseObserver.onCompleted();
return;
}
Trace.Builder builder = Trace.newBuilder()
.setId(streamHeader.getTraceId())
.setUpdate(streamHeader.getUpdate())
.setHeader(checkNotNull(header))
.addAllEntry(entries)
.addAllQuery(queries)
.addAllSharedQueryText(sharedQueryTexts);
if (mainThreadProfile != null) {
builder.setMainThreadProfile(mainThreadProfile);
}
if (auxThreadProfile != null) {
builder.setAuxThreadProfile(auxThreadProfile);
}
trace = builder.build();
} else {
trace = trace.toBuilder()
.addAllSharedQueryText(sharedQueryTexts)
.build();
}
throttledCollectTrace(streamHeader.getAgentId(), streamHeader.getPostV09(), trace,
responseObserver);
}
@RequiresNonNull({"streamHeader", "streamCounts"})
private boolean isEverythingReceived() {
// validate that all data was received, may not receive everything due to gRPC
// maxMessageSize limit exceeded: "Compressed frame exceeds maximum frame size"
if (header == null) {
logger.error("{} - did not receive header, likely due to gRPC maxMessageSize limit"
+ " exceeded", getAgentIdForLogging());
return false;
}
if (sharedQueryTexts.size() < streamCounts.getSharedQueryTextCount()) {
logger.error("{} - expected {} shared query texts, but only received {}, likely due"
+ " to gRPC maxMessageSize limit exceeded for some of them",
getAgentIdForLogging(), streamCounts.getSharedQueryTextCount(),
sharedQueryTexts.size());
return false;
}
if (entries.size() < streamCounts.getEntryCount()) {
logger.error("{} - expected {} entries, but only received {}, likely due to gRPC"
+ " maxMessageSize limit exceeded for some of them", getAgentIdForLogging(),
streamCounts.getEntryCount(), entries.size());
return false;
}
checkState(sharedQueryTexts.size() == streamCounts.getSharedQueryTextCount());
checkState(entries.size() == streamCounts.getEntryCount());
return true;
}
private void logError(Throwable t) {
if (streamHeader == null) {
logger.error(t.getMessage(), t);
} else {
logger.error("{} - {}", getAgentIdForLogging(), t.getMessage(), t);
}
}
@RequiresNonNull("streamHeader")
private String getAgentIdForLogging() {
return grpcCommon.getAgentIdForLogging(streamHeader.getAgentId(),
streamHeader.getPostV09());
}
}
}