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

graphql.normalized.ExecutableNormalizedOperationFactory Maven / Gradle / Ivy

package graphql.normalized;

import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableListMultimap;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.ImmutableSet;
import graphql.GraphQLContext;
import graphql.PublicApi;
import graphql.collect.ImmutableKit;
import graphql.execution.AbortExecutionException;
import graphql.execution.CoercedVariables;
import graphql.execution.MergedField;
import graphql.execution.RawVariables;
import graphql.execution.ValuesResolver;
import graphql.execution.conditional.ConditionalNodes;
import graphql.execution.directives.QueryDirectives;
import graphql.execution.directives.QueryDirectivesImpl;
import graphql.introspection.Introspection;
import graphql.language.Document;
import graphql.language.Field;
import graphql.language.FragmentDefinition;
import graphql.language.FragmentSpread;
import graphql.language.InlineFragment;
import graphql.language.NodeUtil;
import graphql.language.OperationDefinition;
import graphql.language.Selection;
import graphql.language.SelectionSet;
import graphql.language.VariableDefinition;
import graphql.schema.FieldCoordinates;
import graphql.schema.GraphQLCompositeType;
import graphql.schema.GraphQLFieldDefinition;
import graphql.schema.GraphQLInterfaceType;
import graphql.schema.GraphQLNamedOutputType;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLSchema;
import graphql.schema.GraphQLType;
import graphql.schema.GraphQLTypeUtil;
import graphql.schema.GraphQLUnionType;
import graphql.schema.GraphQLUnmodifiedType;
import graphql.schema.impl.SchemaUtil;
import org.jetbrains.annotations.Nullable;

import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Locale;
import java.util.Map;
import java.util.Set;
import java.util.function.BiConsumer;

import static graphql.Assert.assertNotNull;
import static graphql.Assert.assertShouldNeverHappen;
import static graphql.collect.ImmutableKit.map;
import static graphql.schema.GraphQLTypeUtil.unwrapAll;
import static graphql.util.FpKit.filterSet;
import static graphql.util.FpKit.groupingBy;
import static graphql.util.FpKit.intersection;
import static java.util.Collections.singleton;
import static java.util.Collections.singletonList;

/**
 * This factory can create a {@link ExecutableNormalizedOperation} which represents what would be executed
 * during a given graphql operation.
 */
@PublicApi
public class ExecutableNormalizedOperationFactory {
    public static class Options {
        private final GraphQLContext graphQLContext;
        private final Locale locale;
        private final int maxChildrenDepth;

        private Options(GraphQLContext graphQLContext,
                        Locale locale,
                        int maxChildrenDepth) {
            this.graphQLContext = graphQLContext;
            this.locale = locale;
            this.maxChildrenDepth = maxChildrenDepth;
        }

        public static Options defaultOptions() {
            return new Options(
                    GraphQLContext.getDefault(),
                    Locale.getDefault(),
                    Integer.MAX_VALUE);
        }

        /**
         * Locale to use when parsing the query.
         * 

* e.g. can be passed to {@link graphql.schema.Coercing} for parsing. * * @param locale the locale to use * @return new options object to use */ public Options locale(Locale locale) { return new Options(this.graphQLContext, locale, this.maxChildrenDepth); } /** * Context object to use when parsing the operation. *

* Can be used to intercept input values e.g. using {@link graphql.execution.values.InputInterceptor}. * * @param graphQLContext the context to use * @return new options object to use */ public Options graphQLContext(GraphQLContext graphQLContext) { return new Options(graphQLContext, this.locale, this.maxChildrenDepth); } /** * Controls the maximum depth of the operation. Can be used to prevent * against malicious operations. * * @param maxChildrenDepth the max depth * @return new options object to use */ public Options maxChildrenDepth(int maxChildrenDepth) { return new Options(this.graphQLContext, this.locale, maxChildrenDepth); } /** * @return context to use during operation parsing * @see #graphQLContext(GraphQLContext) */ public GraphQLContext getGraphQLContext() { return graphQLContext; } /** * @return locale to use during operation parsing * @see #locale(Locale) */ public Locale getLocale() { return locale; } /** * @return maximum children depth before aborting parsing * @see #maxChildrenDepth(int) */ public int getMaxChildrenDepth() { return maxChildrenDepth; } } private final ConditionalNodes conditionalNodes = new ConditionalNodes(); /** * This will create a runtime representation of the graphql operation that would be executed * in a runtime sense. * * @param graphQLSchema the schema to be used * @param document the {@link Document} holding the operation text * @param operationName the operation name to use * @param coercedVariableValues the coerced variables to use * * @return a runtime representation of the graphql operation. */ public static ExecutableNormalizedOperation createExecutableNormalizedOperation( GraphQLSchema graphQLSchema, Document document, String operationName, CoercedVariables coercedVariableValues ) { NodeUtil.GetOperationResult getOperationResult = NodeUtil.getOperation(document, operationName); return new ExecutableNormalizedOperationFactory().createNormalizedQueryImpl(graphQLSchema, getOperationResult.operationDefinition, getOperationResult.fragmentsByName, coercedVariableValues, null, Options.defaultOptions()); } /** * This will create a runtime representation of the graphql operation that would be executed * in a runtime sense. * * @param graphQLSchema the schema to be used * @param operationDefinition the operation to be executed * @param fragments a set of fragments associated with the operation * @param coercedVariableValues the coerced variables to use * * @return a runtime representation of the graphql operation. */ public static ExecutableNormalizedOperation createExecutableNormalizedOperation(GraphQLSchema graphQLSchema, OperationDefinition operationDefinition, Map fragments, CoercedVariables coercedVariableValues) { return new ExecutableNormalizedOperationFactory().createNormalizedQueryImpl(graphQLSchema, operationDefinition, fragments, coercedVariableValues, null, Options.defaultOptions()); } /** * This will create a runtime representation of the graphql operation that would be executed * in a runtime sense. * * @param graphQLSchema the schema to be used * @param document the {@link Document} holding the operation text * @param operationName the operation name to use * @param rawVariables the raw variables to be coerced * * @return a runtime representation of the graphql operation. */ public static ExecutableNormalizedOperation createExecutableNormalizedOperationWithRawVariables(GraphQLSchema graphQLSchema, Document document, String operationName, RawVariables rawVariables) { return createExecutableNormalizedOperationWithRawVariables(graphQLSchema, document, operationName, rawVariables, Options.defaultOptions()); } /** * This will create a runtime representation of the graphql operation that would be executed * in a runtime sense. * * @param graphQLSchema the schema to be used * @param document the {@link Document} holding the operation text * @param operationName the operation name to use * @param rawVariables the raw variables that have not yet been coerced * @param locale the {@link Locale} to use during coercion * @param graphQLContext the {@link GraphQLContext} to use during coercion * * @return a runtime representation of the graphql operation. */ public static ExecutableNormalizedOperation createExecutableNormalizedOperationWithRawVariables( GraphQLSchema graphQLSchema, Document document, String operationName, RawVariables rawVariables, GraphQLContext graphQLContext, Locale locale ) { return createExecutableNormalizedOperationWithRawVariables( graphQLSchema, document, operationName, rawVariables, Options.defaultOptions().graphQLContext(graphQLContext).locale(locale)); } /** * This will create a runtime representation of the graphql operation that would be executed * in a runtime sense. * * @param graphQLSchema the schema to be used * @param document the {@link Document} holding the operation text * @param operationName the operation name to use * @param rawVariables the raw variables that have not yet been coerced * @param options the {@link Options} to use for parsing * * @return a runtime representation of the graphql operation. */ public static ExecutableNormalizedOperation createExecutableNormalizedOperationWithRawVariables(GraphQLSchema graphQLSchema, Document document, String operationName, RawVariables rawVariables, Options options) { NodeUtil.GetOperationResult getOperationResult = NodeUtil.getOperation(document, operationName); return new ExecutableNormalizedOperationFactory().createExecutableNormalizedOperationImplWithRawVariables(graphQLSchema, getOperationResult.operationDefinition, getOperationResult.fragmentsByName, rawVariables, options ); } private ExecutableNormalizedOperation createExecutableNormalizedOperationImplWithRawVariables(GraphQLSchema graphQLSchema, OperationDefinition operationDefinition, Map fragments, RawVariables rawVariables, Options options) { List variableDefinitions = operationDefinition.getVariableDefinitions(); CoercedVariables coercedVariableValues = ValuesResolver.coerceVariableValues(graphQLSchema, variableDefinitions, rawVariables, options.getGraphQLContext(), options.getLocale()); Map normalizedVariableValues = ValuesResolver.getNormalizedVariableValues(graphQLSchema, variableDefinitions, rawVariables, options.getGraphQLContext(), options.getLocale()); return createNormalizedQueryImpl(graphQLSchema, operationDefinition, fragments, coercedVariableValues, normalizedVariableValues, options); } /** * Creates a new ExecutableNormalizedOperation for the provided query */ private ExecutableNormalizedOperation createNormalizedQueryImpl(GraphQLSchema graphQLSchema, OperationDefinition operationDefinition, Map fragments, CoercedVariables coercedVariableValues, @Nullable Map normalizedVariableValues, Options options) { FieldCollectorNormalizedQueryParams parameters = FieldCollectorNormalizedQueryParams .newParameters() .fragments(fragments) .schema(graphQLSchema) .coercedVariables(coercedVariableValues.toMap()) .normalizedVariables(normalizedVariableValues) .build(); GraphQLObjectType rootType = SchemaUtil.getOperationRootType(graphQLSchema, operationDefinition); CollectNFResult collectFromOperationResult = collectFromOperation(parameters, operationDefinition, rootType); ImmutableListMultimap.Builder fieldToNormalizedField = ImmutableListMultimap.builder(); ImmutableMap.Builder normalizedFieldToMergedField = ImmutableMap.builder(); ImmutableMap.Builder normalizedFieldToQueryDirectives = ImmutableMap.builder(); ImmutableListMultimap.Builder coordinatesToNormalizedFields = ImmutableListMultimap.builder(); BiConsumer captureMergedField = (enf, mergedFld) -> { // QueryDirectivesImpl is a lazy object and only computes itself when asked for QueryDirectives queryDirectives = new QueryDirectivesImpl(mergedFld, graphQLSchema, coercedVariableValues.toMap(), options.getGraphQLContext(), options.getLocale()); normalizedFieldToQueryDirectives.put(enf, queryDirectives); normalizedFieldToMergedField.put(enf, mergedFld); }; for (ExecutableNormalizedField topLevel : collectFromOperationResult.children) { ImmutableList fieldAndAstParents = collectFromOperationResult.normalizedFieldToAstFields.get(topLevel); MergedField mergedField = newMergedField(fieldAndAstParents); captureMergedField.accept(topLevel, mergedField); updateFieldToNFMap(topLevel, fieldAndAstParents, fieldToNormalizedField); updateCoordinatedToNFMap(coordinatesToNormalizedFields, topLevel); buildFieldWithChildren( topLevel, fieldAndAstParents, parameters, fieldToNormalizedField, captureMergedField, coordinatesToNormalizedFields, 1, options.getMaxChildrenDepth()); } for (FieldCollectorNormalizedQueryParams.PossibleMerger possibleMerger : parameters.getPossibleMergerList()) { List childrenWithSameResultKey = possibleMerger.parent.getChildrenWithSameResultKey(possibleMerger.resultKey); ENFMerger.merge(possibleMerger.parent, childrenWithSameResultKey, graphQLSchema); } return new ExecutableNormalizedOperation( operationDefinition.getOperation(), operationDefinition.getName(), new ArrayList<>(collectFromOperationResult.children), fieldToNormalizedField.build(), normalizedFieldToMergedField.build(), normalizedFieldToQueryDirectives.build(), coordinatesToNormalizedFields.build() ); } private void buildFieldWithChildren(ExecutableNormalizedField executableNormalizedField, ImmutableList fieldAndAstParents, FieldCollectorNormalizedQueryParams fieldCollectorNormalizedQueryParams, ImmutableListMultimap.Builder fieldNormalizedField, BiConsumer captureMergedField, ImmutableListMultimap.Builder coordinatesToNormalizedFields, int curLevel, int maxLevel) { if (curLevel > maxLevel) { throw new AbortExecutionException("Maximum query depth exceeded " + curLevel + " > " + maxLevel); } CollectNFResult nextLevel = collectFromMergedField(fieldCollectorNormalizedQueryParams, executableNormalizedField, fieldAndAstParents, curLevel + 1); for (ExecutableNormalizedField childENF : nextLevel.children) { executableNormalizedField.addChild(childENF); ImmutableList childFieldAndAstParents = nextLevel.normalizedFieldToAstFields.get(childENF); MergedField mergedField = newMergedField(childFieldAndAstParents); captureMergedField.accept(childENF, mergedField); updateFieldToNFMap(childENF, childFieldAndAstParents, fieldNormalizedField); updateCoordinatedToNFMap(coordinatesToNormalizedFields, childENF); buildFieldWithChildren(childENF, childFieldAndAstParents, fieldCollectorNormalizedQueryParams, fieldNormalizedField, captureMergedField, coordinatesToNormalizedFields, curLevel + 1, maxLevel); } } private static MergedField newMergedField(ImmutableList fieldAndAstParents) { return MergedField.newMergedField(map(fieldAndAstParents, fieldAndAstParent -> fieldAndAstParent.field)).build(); } private void updateFieldToNFMap(ExecutableNormalizedField executableNormalizedField, ImmutableList mergedField, ImmutableListMultimap.Builder fieldToNormalizedField) { for (FieldAndAstParent astField : mergedField) { fieldToNormalizedField.put(astField.field, executableNormalizedField); } } private void updateCoordinatedToNFMap(ImmutableListMultimap.Builder coordinatesToNormalizedFields, ExecutableNormalizedField topLevel) { for (String objectType : topLevel.getObjectTypeNames()) { FieldCoordinates coordinates = FieldCoordinates.coordinates(objectType, topLevel.getFieldName()); coordinatesToNormalizedFields.put(coordinates, topLevel); } } private static class FieldAndAstParent { final Field field; final GraphQLCompositeType astParentType; private FieldAndAstParent(Field field, GraphQLCompositeType astParentType) { this.field = field; this.astParentType = astParentType; } } public static class CollectNFResult { private final Collection children; private final ImmutableListMultimap normalizedFieldToAstFields; public CollectNFResult(Collection children, ImmutableListMultimap normalizedFieldToAstFields) { this.children = children; this.normalizedFieldToAstFields = normalizedFieldToAstFields; } } public CollectNFResult collectFromMergedField(FieldCollectorNormalizedQueryParams parameters, ExecutableNormalizedField executableNormalizedField, ImmutableList mergedField, int level) { List fieldDefs = executableNormalizedField.getFieldDefinitions(parameters.getGraphQLSchema()); Set possibleObjects = resolvePossibleObjects(fieldDefs, parameters.getGraphQLSchema()); if (possibleObjects.isEmpty()) { return new CollectNFResult(ImmutableKit.emptyList(), ImmutableListMultimap.of()); } List collectedFields = new ArrayList<>(); for (FieldAndAstParent fieldAndAstParent : mergedField) { if (fieldAndAstParent.field.getSelectionSet() == null) { continue; } GraphQLFieldDefinition fieldDefinition = Introspection.getFieldDef(parameters.getGraphQLSchema(), fieldAndAstParent.astParentType, fieldAndAstParent.field.getName()); GraphQLUnmodifiedType astParentType = unwrapAll(fieldDefinition.getType()); this.collectFromSelectionSet(parameters, fieldAndAstParent.field.getSelectionSet(), collectedFields, (GraphQLCompositeType) astParentType, possibleObjects ); } Map> fieldsByName = fieldsByResultKey(collectedFields); ImmutableList.Builder resultNFs = ImmutableList.builder(); ImmutableListMultimap.Builder normalizedFieldToAstFields = ImmutableListMultimap.builder(); createNFs(resultNFs, parameters, fieldsByName, normalizedFieldToAstFields, level, executableNormalizedField); return new CollectNFResult(resultNFs.build(), normalizedFieldToAstFields.build()); } private Map> fieldsByResultKey(List collectedFields) { Map> fieldsByName = new LinkedHashMap<>(); for (CollectedField collectedField : collectedFields) { fieldsByName.computeIfAbsent(collectedField.field.getResultKey(), ignored -> new ArrayList<>()).add(collectedField); } return fieldsByName; } public CollectNFResult collectFromOperation(FieldCollectorNormalizedQueryParams parameters, OperationDefinition operationDefinition, GraphQLObjectType rootType) { Set possibleObjects = ImmutableSet.of(rootType); List collectedFields = new ArrayList<>(); collectFromSelectionSet(parameters, operationDefinition.getSelectionSet(), collectedFields, rootType, possibleObjects); // group by result key Map> fieldsByName = fieldsByResultKey(collectedFields); ImmutableList.Builder resultNFs = ImmutableList.builder(); ImmutableListMultimap.Builder normalizedFieldToAstFields = ImmutableListMultimap.builder(); createNFs(resultNFs, parameters, fieldsByName, normalizedFieldToAstFields, 1, null); return new CollectNFResult(resultNFs.build(), normalizedFieldToAstFields.build()); } private void createNFs(ImmutableList.Builder nfListBuilder, FieldCollectorNormalizedQueryParams parameters, Map> fieldsByName, ImmutableListMultimap.Builder normalizedFieldToAstFields, int level, ExecutableNormalizedField parent) { for (String resultKey : fieldsByName.keySet()) { List fieldsWithSameResultKey = fieldsByName.get(resultKey); List commonParentsGroups = groupByCommonParents(fieldsWithSameResultKey); for (CollectedFieldGroup fieldGroup : commonParentsGroups) { ExecutableNormalizedField nf = createNF(parameters, fieldGroup, level, parent); if (nf == null) { continue; } for (CollectedField collectedField : fieldGroup.fields) { normalizedFieldToAstFields.put(nf, new FieldAndAstParent(collectedField.field, collectedField.astTypeCondition)); } nfListBuilder.add(nf); } if (commonParentsGroups.size() > 1) { parameters.addPossibleMergers(parent, resultKey); } } } private ExecutableNormalizedField createNF(FieldCollectorNormalizedQueryParams parameters, CollectedFieldGroup collectedFieldGroup, int level, ExecutableNormalizedField parent) { Field field; Set objectTypes = collectedFieldGroup.objectTypes; field = collectedFieldGroup.fields.iterator().next().field; String fieldName = field.getName(); GraphQLFieldDefinition fieldDefinition = Introspection.getFieldDef(parameters.getGraphQLSchema(), objectTypes.iterator().next(), fieldName); Map argumentValues = ValuesResolver.getArgumentValues(fieldDefinition.getArguments(), field.getArguments(), CoercedVariables.of(parameters.getCoercedVariableValues()), parameters.getGraphQLContext(), parameters.getLocale()); Map normalizedArgumentValues = null; if (parameters.getNormalizedVariableValues() != null) { normalizedArgumentValues = ValuesResolver.getNormalizedArgumentValues(fieldDefinition.getArguments(), field.getArguments(), parameters.getNormalizedVariableValues()); } ImmutableList objectTypeNames = map(objectTypes, GraphQLObjectType::getName); return ExecutableNormalizedField.newNormalizedField() .alias(field.getAlias()) .resolvedArguments(argumentValues) .normalizedArguments(normalizedArgumentValues) .astArguments(field.getArguments()) .objectTypeNames(objectTypeNames) .fieldName(fieldName) .level(level) .parent(parent) .build(); } private static class CollectedFieldGroup { Set objectTypes; Set fields; public CollectedFieldGroup(Set fields, Set objectTypes) { this.fields = fields; this.objectTypes = objectTypes; } } private List groupByCommonParents(Collection fields) { ImmutableSet.Builder objectTypes = ImmutableSet.builder(); for (CollectedField collectedField : fields) { objectTypes.addAll(collectedField.objectTypes); } Set allRelevantObjects = objectTypes.build(); Map> groupByAstParent = groupingBy(fields, fieldAndType -> fieldAndType.astTypeCondition); if (groupByAstParent.size() == 1) { return singletonList(new CollectedFieldGroup(ImmutableSet.copyOf(fields), allRelevantObjects)); } ImmutableList.Builder result = ImmutableList.builder(); for (GraphQLObjectType objectType : allRelevantObjects) { Set relevantFields = filterSet(fields, field -> field.objectTypes.contains(objectType)); result.add(new CollectedFieldGroup(relevantFields, singleton(objectType))); } return result.build(); } private void collectFromSelectionSet(FieldCollectorNormalizedQueryParams parameters, SelectionSet selectionSet, List result, GraphQLCompositeType astTypeCondition, Set possibleObjects ) { for (Selection selection : selectionSet.getSelections()) { if (selection instanceof Field) { collectField(parameters, result, (Field) selection, possibleObjects, astTypeCondition); } else if (selection instanceof InlineFragment) { collectInlineFragment(parameters, result, (InlineFragment) selection, possibleObjects, astTypeCondition); } else if (selection instanceof FragmentSpread) { collectFragmentSpread(parameters, result, (FragmentSpread) selection, possibleObjects); } } } private static class CollectedField { Field field; Set objectTypes; GraphQLCompositeType astTypeCondition; public CollectedField(Field field, Set objectTypes, GraphQLCompositeType astTypeCondition) { this.field = field; this.objectTypes = objectTypes; this.astTypeCondition = astTypeCondition; } public boolean isAbstract() { return GraphQLTypeUtil.isInterfaceOrUnion(astTypeCondition); } public boolean isConcrete() { return GraphQLTypeUtil.isObjectType(astTypeCondition); } } private void collectFragmentSpread(FieldCollectorNormalizedQueryParams parameters, List result, FragmentSpread fragmentSpread, Set possibleObjects ) { if (!conditionalNodes.shouldInclude(fragmentSpread, parameters.getCoercedVariableValues(), parameters.getGraphQLSchema(), parameters.getGraphQLContext())) { return; } FragmentDefinition fragmentDefinition = assertNotNull(parameters.getFragmentsByName().get(fragmentSpread.getName())); if (!conditionalNodes.shouldInclude(fragmentDefinition, parameters.getCoercedVariableValues(), parameters.getGraphQLSchema(), parameters.getGraphQLContext())) { return; } GraphQLCompositeType newAstTypeCondition = (GraphQLCompositeType) assertNotNull(parameters.getGraphQLSchema().getType(fragmentDefinition.getTypeCondition().getName())); Set newPossibleObjects = narrowDownPossibleObjects(possibleObjects, newAstTypeCondition, parameters.getGraphQLSchema()); collectFromSelectionSet(parameters, fragmentDefinition.getSelectionSet(), result, newAstTypeCondition, newPossibleObjects); } private void collectInlineFragment(FieldCollectorNormalizedQueryParams parameters, List result, InlineFragment inlineFragment, Set possibleObjects, GraphQLCompositeType astTypeCondition ) { if (!conditionalNodes.shouldInclude(inlineFragment, parameters.getCoercedVariableValues(), parameters.getGraphQLSchema(), parameters.getGraphQLContext())) { return; } Set newPossibleObjects = possibleObjects; GraphQLCompositeType newAstTypeCondition = astTypeCondition; if (inlineFragment.getTypeCondition() != null) { newAstTypeCondition = (GraphQLCompositeType) parameters.getGraphQLSchema().getType(inlineFragment.getTypeCondition().getName()); newPossibleObjects = narrowDownPossibleObjects(possibleObjects, newAstTypeCondition, parameters.getGraphQLSchema()); } collectFromSelectionSet(parameters, inlineFragment.getSelectionSet(), result, newAstTypeCondition, newPossibleObjects); } private void collectField(FieldCollectorNormalizedQueryParams parameters, List result, Field field, Set possibleObjectTypes, GraphQLCompositeType astTypeCondition ) { if (!conditionalNodes.shouldInclude(field, parameters.getCoercedVariableValues(), parameters.getGraphQLSchema(), parameters.getGraphQLContext())) { return; } // this means there is actually no possible type for this field, and we are done if (possibleObjectTypes.isEmpty()) { return; } result.add(new CollectedField(field, possibleObjectTypes, astTypeCondition)); } private Set narrowDownPossibleObjects(Set currentOnes, GraphQLCompositeType typeCondition, GraphQLSchema graphQLSchema) { ImmutableSet resolvedTypeCondition = resolvePossibleObjects(typeCondition, graphQLSchema); if (currentOnes.isEmpty()) { return resolvedTypeCondition; } // Faster intersection, as either set often has a size of 1. return intersection(currentOnes, resolvedTypeCondition); } private ImmutableSet resolvePossibleObjects(List defs, GraphQLSchema graphQLSchema) { ImmutableSet.Builder builder = ImmutableSet.builder(); for (GraphQLFieldDefinition def : defs) { GraphQLUnmodifiedType outputType = unwrapAll(def.getType()); if (outputType instanceof GraphQLCompositeType) { builder.addAll(resolvePossibleObjects((GraphQLCompositeType) outputType, graphQLSchema)); } } return builder.build(); } private ImmutableSet resolvePossibleObjects(GraphQLCompositeType type, GraphQLSchema graphQLSchema) { if (type instanceof GraphQLObjectType) { return ImmutableSet.of((GraphQLObjectType) type); } else if (type instanceof GraphQLInterfaceType) { return ImmutableSet.copyOf(graphQLSchema.getImplementations((GraphQLInterfaceType) type)); } else if (type instanceof GraphQLUnionType) { List unionTypes = ((GraphQLUnionType) type).getTypes(); return ImmutableSet.copyOf(ImmutableKit.map(unionTypes, GraphQLObjectType.class::cast)); } else { return assertShouldNeverHappen(); } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy