graphql.execution.Execution Maven / Gradle / Ivy
package graphql.execution;
import graphql.ExecutionInput;
import graphql.ExecutionResult;
import graphql.ExecutionResultImpl;
import graphql.Internal;
import graphql.MutationNotSupportedError;
import graphql.execution.instrumentation.Instrumentation;
import graphql.execution.instrumentation.InstrumentationContext;
import graphql.execution.instrumentation.InstrumentationState;
import graphql.execution.instrumentation.parameters.InstrumentationDataFetchParameters;
import graphql.language.Document;
import graphql.language.Field;
import graphql.language.OperationDefinition;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLSchema;
import java.util.Collections;
import java.util.List;
import java.util.Map;
import java.util.concurrent.CompletableFuture;
import static graphql.Assert.assertShouldNeverHappen;
import static graphql.execution.ExecutionStrategyParameters.newParameters;
import static graphql.execution.ExecutionTypeInfo.newTypeInfo;
import static graphql.language.OperationDefinition.Operation.MUTATION;
import static graphql.language.OperationDefinition.Operation.QUERY;
import static graphql.language.OperationDefinition.Operation.SUBSCRIPTION;
import static java.util.concurrent.CompletableFuture.completedFuture;
@Internal
public class Execution {
private final FieldCollector fieldCollector = new FieldCollector();
private final ExecutionStrategy queryStrategy;
private final ExecutionStrategy mutationStrategy;
private final ExecutionStrategy subscriptionStrategy;
private final Instrumentation instrumentation;
public Execution(ExecutionStrategy queryStrategy, ExecutionStrategy mutationStrategy, ExecutionStrategy subscriptionStrategy, Instrumentation instrumentation) {
this.queryStrategy = queryStrategy != null ? queryStrategy : new AsyncExecutionStrategy();
this.mutationStrategy = mutationStrategy != null ? mutationStrategy : new AsyncSerialExecutionStrategy();
this.subscriptionStrategy = subscriptionStrategy != null ? subscriptionStrategy : new AsyncExecutionStrategy();
this.instrumentation = instrumentation;
}
public CompletableFuture execute(Document document, GraphQLSchema graphQLSchema, ExecutionId executionId, ExecutionInput executionInput, InstrumentationState instrumentationState) {
ExecutionContext executionContext = new ExecutionContextBuilder()
.valuesResolver(new ValuesResolver())
.instrumentation(instrumentation)
.instrumentationState(instrumentationState)
.executionId(executionId)
.graphQLSchema(graphQLSchema)
.queryStrategy(queryStrategy)
.mutationStrategy(mutationStrategy)
.subscriptionStrategy(subscriptionStrategy)
.context(executionInput.getContext())
.root(executionInput.getRoot())
.document(document)
.operationName(executionInput.getOperationName())
.variables(executionInput.getVariables())
.build();
return executeOperation(executionContext, executionInput.getRoot(), executionContext.getOperationDefinition());
}
private CompletableFuture executeOperation(ExecutionContext executionContext, Object root, OperationDefinition operationDefinition) {
InstrumentationContext dataFetchCtx = instrumentation.beginDataFetch(new InstrumentationDataFetchParameters(executionContext));
OperationDefinition.Operation operation = operationDefinition.getOperation();
GraphQLObjectType operationRootType = getOperationRootType(executionContext.getGraphQLSchema(), operation);
//
// do we have a mutation operation root type. The spec says if we don't then mutations are not allowed to be executed
//
// for the record earlier code has asserted that we have a query type in the schema since the spec says this is
// ALWAYS required
if (operation == MUTATION && operationRootType == null) {
return completedFuture(new ExecutionResultImpl(Collections.singletonList(new MutationNotSupportedError())));
}
FieldCollectorParameters collectorParameters = FieldCollectorParameters.newParameters()
.schema(executionContext.getGraphQLSchema())
.objectType(operationRootType)
.fragments(executionContext.getFragmentsByName())
.variables(executionContext.getVariables())
.build();
Map> fields = fieldCollector.collectFields(collectorParameters, operationDefinition.getSelectionSet());
ExecutionPath path = ExecutionPath.rootPath();
ExecutionTypeInfo typeInfo = newTypeInfo().type(operationRootType).path(path).build();
NonNullableFieldValidator nonNullableFieldValidator = new NonNullableFieldValidator(executionContext, typeInfo);
ExecutionStrategyParameters parameters = newParameters()
.typeInfo(typeInfo)
.source(root)
.fields(fields)
.nonNullFieldValidator(nonNullableFieldValidator)
.path(path)
.build();
CompletableFuture result;
try {
if (operation == OperationDefinition.Operation.MUTATION) {
result = mutationStrategy.execute(executionContext, parameters);
} else if (operation == SUBSCRIPTION) {
result = subscriptionStrategy.execute(executionContext, parameters);
} else {
result = queryStrategy.execute(executionContext, parameters);
}
} catch (NonNullableFieldWasNullException e) {
// this means it was non null types all the way from an offending non null type
// up to the root object type and there was a a null value some where.
//
// The spec says we should return null for the data in this case
//
// http://facebook.github.io/graphql/#sec-Errors-and-Non-Nullability
//
result = completedFuture(new ExecutionResultImpl(null, executionContext.getErrors()));
}
result = result.whenComplete(dataFetchCtx::onEnd);
return result;
}
private GraphQLObjectType getOperationRootType(GraphQLSchema graphQLSchema, OperationDefinition.Operation operation) {
if (operation == MUTATION) {
return graphQLSchema.getMutationType();
} else if (operation == QUERY) {
return graphQLSchema.getQueryType();
} else if (operation == SUBSCRIPTION) {
return graphQLSchema.getSubscriptionType();
} else {
return assertShouldNeverHappen("Unhandled case. An extra operation enum has been added without code support");
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy