graphql.agent.result.ExecutionTrackingResult Maven / Gradle / Ivy
package graphql.agent.result;
import graphql.PublicApi;
import graphql.execution.ResultPath;
import org.dataloader.DataLoader;
import java.util.ArrayList;
import java.util.List;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicLong;
import java.util.concurrent.atomic.AtomicReference;
import static graphql.agent.result.ExecutionTrackingResult.DFResultType.PENDING;
/**
* This is the result of the agent tracking an execution.
* It can be found inside the GraphQLContext after the execution with the key {@link ExecutionTrackingResult#EXECUTION_TRACKING_KEY}
*
* Note: While this is public API, the main goal is temporary debugging to understand an execution better with minimal overhead.
* Therefore this will evolve over time if needed to be performant and reflect the overall execution.
* It is not recommended to have the agent on always or to rely on this class during normal execution
*/
@PublicApi
public class ExecutionTrackingResult {
public static final String EXECUTION_TRACKING_KEY = "__GJ_AGENT_EXECUTION_TRACKING";
public final AtomicReference startThread = new AtomicReference<>();
public final AtomicReference endThread = new AtomicReference<>();
public final AtomicLong startExecutionTime = new AtomicLong();
public final AtomicLong endExecutionTime = new AtomicLong();
public final Map resultPathToDataLoaderUsed = new ConcurrentHashMap<>();
public final Map dataLoaderToName = new ConcurrentHashMap<>();
public final Map timePerPath = new ConcurrentHashMap<>();
public final Map finishedTimePerPath = new ConcurrentHashMap<>();
public final Map finishedThreadPerPath = new ConcurrentHashMap<>();
public final Map startInvocationThreadPerPath = new ConcurrentHashMap<>();
private final Map dfResultTypes = new ConcurrentHashMap<>();
public final Map> dataLoaderNameToBatchCall = new ConcurrentHashMap<>();
public static class BatchLoadingCall {
public BatchLoadingCall(int keyCount, String threadName) {
this.keyCount = keyCount;
this.threadName = threadName;
}
public final int keyCount;
public final String threadName;
}
public String print(String executionId) {
StringBuilder s = new StringBuilder();
s.append("==========================").append("\n");
s.append("Summary for execution with id ").append(executionId).append("\n");
s.append("==========================").append("\n");
s.append("Execution time in ms:").append((endExecutionTime.get() - startExecutionTime.get()) / 1_000_000L).append("\n");
s.append("Fields count: ").append(timePerPath.keySet().size()).append("\n");
s.append("Blocking fields count: ").append(dfResultTypes.values().stream().filter(dfResultType -> dfResultType != PENDING).count()).append("\n");
s.append("Nonblocking fields count: ").append(dfResultTypes.values().stream().filter(dfResultType -> dfResultType == PENDING).count()).append("\n");
s.append("DataLoaders used: ").append(dataLoaderToName.size()).append("\n");
s.append("DataLoader names: ").append(dataLoaderToName.values()).append("\n");
s.append("start execution thread: '").append(startThread.get()).append("'\n");
s.append("end execution thread: '").append(endThread.get()).append("'\n");
s.append("BatchLoader calls details: ").append("\n");
s.append("==========================").append("\n");
for (String dataLoaderName : dataLoaderNameToBatchCall.keySet()) {
s.append("Batch call: '").append(dataLoaderName).append("' made ").append(dataLoaderNameToBatchCall.get(dataLoaderName).size()).append(" times, ").append("\n");
for (BatchLoadingCall batchLoadingCall : dataLoaderNameToBatchCall.get(dataLoaderName)) {
s.append("Batch call with ").append(batchLoadingCall.keyCount).append(" keys ").append(" in thread ").append(batchLoadingCall.threadName).append("\n");
}
List resultPathUsed = new ArrayList<>();
for (ResultPath resultPath : resultPathToDataLoaderUsed.keySet()) {
if (resultPathToDataLoaderUsed.get(resultPath).equals(dataLoaderName)) {
resultPathUsed.add(resultPath);
}
}
s.append("DataLoader: '").append(dataLoaderName).append("' used in fields: ").append(resultPathUsed).append("\n");
}
s.append("Field details:").append("\n");
s.append("===============").append("\n");
for (ResultPath path : timePerPath.keySet()) {
s.append("Field: '").append(path).append("'\n");
s.append("invocation time: ").append(timePerPath.get(path)).append(" nano seconds, ").append("\n");
s.append("completion time: ").append(finishedTimePerPath.get(path)).append(" nano seconds, ").append("\n");
s.append("Result type: ").append(dfResultTypes.get(path)).append("\n");
s.append("invoked in thread: ").append(startInvocationThreadPerPath.get(path)).append("\n");
s.append("finished in thread: ").append(finishedThreadPerPath.get(path)).append("\n");
s.append("-------------\n");
}
s.append("==========================").append("\n");
s.append("==========================").append("\n");
return s.toString();
}
@Override
public String toString() {
return "ExecutionData{" +
"resultPathToDataLoaderUsed=" + resultPathToDataLoaderUsed +
", dataLoaderNames=" + dataLoaderToName.values() +
", timePerPath=" + timePerPath +
", dfResultTypes=" + dfResultTypes +
'}';
}
public enum DFResultType {
DONE_OK,
DONE_EXCEPTIONALLY,
DONE_CANCELLED,
PENDING,
}
public List getDataLoaderNames() {
return new ArrayList<>(dataLoaderToName.values());
}
public void start(ResultPath path, long startTime) {
timePerPath.put(path, startTime);
}
public void end(ResultPath path, long endTime) {
timePerPath.put(path, endTime - timePerPath.get(path));
}
public int dataFetcherCount() {
return timePerPath.size();
}
public long getTime(ResultPath path) {
return timePerPath.get(path);
}
public long getTime(String path) {
return timePerPath.get(ResultPath.parse(path));
}
public void setDfResultTypes(ResultPath resultPath, DFResultType resultTypes) {
dfResultTypes.put(resultPath, resultTypes);
}
public DFResultType getDfResultTypes(ResultPath resultPath) {
return dfResultTypes.get(resultPath);
}
public DFResultType getDfResultTypes(String resultPath) {
return dfResultTypes.get(ResultPath.parse(resultPath));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy