All Downloads are FREE. Search and download functionalities are using the official Maven repository.

graphql.execution.ExecutionStepInfo Maven / Gradle / Ivy

There is a newer version: 230521-nf-execution
Show newest version
package graphql.execution;

import graphql.PublicApi;
import graphql.schema.GraphQLFieldDefinition;
import graphql.schema.GraphQLInterfaceType;
import graphql.schema.GraphQLNonNull;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLOutputType;
import graphql.schema.GraphQLTypeUtil;

import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.function.Consumer;

import static graphql.Assert.assertNotNull;
import static graphql.Assert.assertTrue;
import static graphql.schema.GraphQLTypeUtil.isList;

/**
 * As the graphql query executes, it forms a hierarchy from parent fields (and their type) to their child fields (and their type)
 * until a scalar type is encountered; this class captures that execution type information.
 * 

* The static graphql type system (rightly) does not contain a hierarchy of child to parent types nor the nonnull ness of * type instances, so this helper class adds this information during query execution. */ @PublicApi public class ExecutionStepInfo { private final GraphQLOutputType type; private final ExecutionPath path; private final ExecutionStepInfo parent; // field, fieldDefinition, fieldContainer and arguments stay the same for steps inside a list field private final MergedField field; private final GraphQLFieldDefinition fieldDefinition; private final GraphQLObjectType fieldContainer; private final Map arguments; private ExecutionStepInfo(GraphQLOutputType type, GraphQLFieldDefinition fieldDefinition, MergedField field, ExecutionPath path, ExecutionStepInfo parent, Map arguments, GraphQLObjectType fieldsContainer) { this.fieldDefinition = fieldDefinition; this.field = field; this.path = path; this.parent = parent; this.type = assertNotNull(type, "you must provide a graphql type"); this.arguments = arguments; this.fieldContainer = fieldsContainer; } /** * The GraphQLObjectType where fieldDefinition is defined. * Note: * For the Introspection field __typename the returned object type doesn't actually contain the fieldDefinition. * * @return GraphQLObjectType defining {@link #getFieldDefinition()} */ public GraphQLObjectType getFieldContainer() { return fieldContainer; } /** * This returns the type for the current step. * * @return the graphql type in question */ public GraphQLOutputType getType() { return type; } /** * This returns the type which is unwrapped if it was {@link GraphQLNonNull} wrapped * * @return the graphql type in question */ public GraphQLOutputType getUnwrappedNonNullType() { return (GraphQLOutputType) GraphQLTypeUtil.unwrapNonNull(this.type); } /** * This returns the field definition that is in play when this type info was created or null * if the type is a root query type * * @return the field definition or null if there is not one */ public GraphQLFieldDefinition getFieldDefinition() { return fieldDefinition; } /** * This returns the AST fields that matches the {@link #getFieldDefinition()} during execution * * @return the merged fields */ public MergedField getField() { return field; } /** * @return the {@link ExecutionPath} to this info */ public ExecutionPath getPath() { return path; } /** * @return true if the type must be nonnull */ public boolean isNonNullType() { return GraphQLTypeUtil.isNonNull(this.type); } /** * @return true if the type is a list */ public boolean isListType() { return isList(type); } /** * @return the resolved arguments that have been passed to this field */ public Map getArguments() { return new LinkedHashMap<>(arguments); } /** * Returns the named argument * * @param name the name of the argument * @param you decide what type it is * * @return the named argument or null if its not present */ @SuppressWarnings("unchecked") public T getArgument(String name) { return (T) arguments.get(name); } /** * @return the parent type information */ public ExecutionStepInfo getParent() { return parent; } /** * @return true if the type has a parent (most do) */ public boolean hasParent() { return parent != null; } /** * This allows you to morph a type into a more specialized form yet return the same * parent and non-null ness, for example taking a {@link GraphQLInterfaceType} * and turning it into a specific {@link graphql.schema.GraphQLObjectType} * after type resolution has occurred * * @param newType the new type to be * * @return a new type info with the same */ public ExecutionStepInfo changeTypeWithPreservedNonNull(GraphQLOutputType newType) { assertTrue(!GraphQLTypeUtil.isNonNull(newType), "newType can't be non null"); if (isNonNullType()) { return new ExecutionStepInfo(GraphQLNonNull.nonNull(newType), fieldDefinition, field, path, this.parent, arguments, this.fieldContainer); } else { return new ExecutionStepInfo(newType, fieldDefinition, field, path, this.parent, arguments, this.fieldContainer); } } /** * @return the type in graphql SDL format, eg [typeName!]! */ public String simplePrint() { return GraphQLTypeUtil.simplePrint(type); } @Override public String toString() { return "ExecutionStepInfo{" + " path=" + path + ", type=" + type + ", parent=" + parent + ", fieldDefinition=" + fieldDefinition + '}'; } public ExecutionStepInfo transform(Consumer builderConsumer) { Builder builder = new Builder(this); builderConsumer.accept(builder); return builder.build(); } public String getResultKey() { return field.getResultKey(); } /** * @return a builder of type info */ public static ExecutionStepInfo.Builder newExecutionStepInfo() { return new Builder(); } public static ExecutionStepInfo.Builder newExecutionStepInfo(ExecutionStepInfo existing) { return new Builder(existing); } public static class Builder { GraphQLOutputType type; ExecutionStepInfo parentInfo; GraphQLFieldDefinition fieldDefinition; GraphQLObjectType fieldContainer; MergedField field; ExecutionPath path; Map arguments = new LinkedHashMap<>(); /** * @see ExecutionStepInfo#newExecutionStepInfo() */ private Builder() { } private Builder(ExecutionStepInfo existing) { this.type = existing.type; this.parentInfo = existing.parent; this.fieldDefinition = existing.fieldDefinition; this.field = existing.field; this.path = existing.path; this.arguments = existing.getArguments(); } public Builder type(GraphQLOutputType type) { this.type = type; return this; } public Builder parentInfo(ExecutionStepInfo executionStepInfo) { this.parentInfo = executionStepInfo; return this; } public Builder fieldDefinition(GraphQLFieldDefinition fieldDefinition) { this.fieldDefinition = fieldDefinition; return this; } public Builder field(MergedField field) { this.field = field; return this; } public Builder path(ExecutionPath executionPath) { this.path = executionPath; return this; } public Builder arguments(Map arguments) { this.arguments = new LinkedHashMap<>(arguments == null ? Collections.emptyMap() : arguments); return this; } public Builder fieldContainer(GraphQLObjectType fieldContainer) { this.fieldContainer = fieldContainer; return this; } public ExecutionStepInfo build() { return new ExecutionStepInfo(type, fieldDefinition, field, path, parentInfo, arguments, fieldContainer); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy