graphql.normalized.NormalizedQueryFactory 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 com.google.common.collect.LinkedHashMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Sets;
import graphql.Assert;
import graphql.Internal;
import graphql.execution.ConditionalNodes;
import graphql.execution.MergedField;
import graphql.execution.ValuesResolver;
import graphql.execution.nextgen.Common;
import graphql.introspection.Introspection;
import graphql.language.Argument;
import graphql.language.AstComparator;
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.GraphQLObjectType;
import graphql.schema.GraphQLSchema;
import graphql.schema.GraphQLUnionType;
import graphql.schema.GraphQLUnmodifiedType;
import org.jetbrains.annotations.Nullable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import static graphql.Assert.assertNotNull;
import static graphql.collect.ImmutableKit.map;
import static graphql.schema.GraphQLTypeUtil.simplePrint;
import static graphql.schema.GraphQLTypeUtil.unwrapAll;
@Internal
public class NormalizedQueryFactory {
private final ValuesResolver valuesResolver = new ValuesResolver();
private final ConditionalNodes conditionalNodes = new ConditionalNodes();
public static NormalizedQuery createNormalizedQuery(GraphQLSchema graphQLSchema,
Document document,
String operationName,
Map coercedVariableValues) {
NodeUtil.GetOperationResult getOperationResult = NodeUtil.getOperation(document, operationName);
return new NormalizedQueryFactory().createNormalizedQueryImpl(graphQLSchema, getOperationResult.operationDefinition, getOperationResult.fragmentsByName, coercedVariableValues, null);
}
public static NormalizedQuery createNormalizedQuery(GraphQLSchema graphQLSchema,
OperationDefinition operationDefinition,
Map fragments,
Map coercedVariableValues) {
return new NormalizedQueryFactory().createNormalizedQueryImpl(graphQLSchema, operationDefinition, fragments, coercedVariableValues, null);
}
public static NormalizedQuery createNormalizedQueryWithRawVariables(GraphQLSchema graphQLSchema,
Document document,
String operationName,
Map rawVariables) {
NodeUtil.GetOperationResult getOperationResult = NodeUtil.getOperation(document, operationName);
return new NormalizedQueryFactory().createNormalizedQueryImplWithRawVariables(graphQLSchema, getOperationResult.operationDefinition, getOperationResult.fragmentsByName, rawVariables);
}
private NormalizedQuery createNormalizedQueryImplWithRawVariables(GraphQLSchema graphQLSchema,
OperationDefinition operationDefinition,
Map fragments,
Map rawVariables
) {
List variableDefinitions = operationDefinition.getVariableDefinitions();
Map coerceVariableValues = valuesResolver.coerceVariableValues(graphQLSchema, variableDefinitions, rawVariables);
Map normalizedVariableValues = valuesResolver.getNormalizedVariableValues(graphQLSchema, variableDefinitions, rawVariables);
return createNormalizedQueryImpl(graphQLSchema, operationDefinition, fragments, coerceVariableValues, normalizedVariableValues);
}
/**
* Creates a new Normalized query tree for the provided query
*/
private NormalizedQuery createNormalizedQueryImpl(GraphQLSchema graphQLSchema,
OperationDefinition operationDefinition,
Map fragments,
Map coercedVariableValues,
@Nullable Map normalizedVariableValues) {
FieldCollectorNormalizedQueryParams parameters = FieldCollectorNormalizedQueryParams
.newParameters()
.fragments(fragments)
.schema(graphQLSchema)
.coercedVariables(coercedVariableValues)
.normalizedVariables(normalizedVariableValues)
.build();
GraphQLObjectType rootType = Common.getOperationRootType(graphQLSchema, operationDefinition);
CollectFieldResult collectFromOperationResult = collectFromOperation(parameters, operationDefinition, rootType);
ImmutableListMultimap.Builder fieldToNormalizedField = ImmutableListMultimap.builder();
ImmutableMap.Builder normalizedFieldToMergedField = ImmutableMap.builder();
ImmutableListMultimap.Builder coordinatesToNormalizedFields = ImmutableListMultimap.builder();
for (NormalizedField topLevel : collectFromOperationResult.children) {
ImmutableList mergedField = collectFromOperationResult.normalizedFieldToAstFields.get(topLevel);
normalizedFieldToMergedField.put(topLevel, MergedField.newMergedField(mergedField).build());
updateFieldToNFMap(topLevel, mergedField, fieldToNormalizedField);
updateCoordinatedToNFMap(coordinatesToNormalizedFields, topLevel);
buildFieldWithChildren(topLevel,
mergedField,
parameters,
fieldToNormalizedField,
normalizedFieldToMergedField,
coordinatesToNormalizedFields,
1);
}
return new NormalizedQuery(new ArrayList<>(collectFromOperationResult.children), fieldToNormalizedField.build(), normalizedFieldToMergedField.build(), coordinatesToNormalizedFields.build());
}
private void buildFieldWithChildren(NormalizedField field,
ImmutableList mergedField,
FieldCollectorNormalizedQueryParams fieldCollectorNormalizedQueryParams,
ImmutableListMultimap.Builder fieldNormalizedField,
ImmutableMap.Builder normalizedFieldToMergedField,
ImmutableListMultimap.Builder coordinatesToNormalizedFields,
int curLevel) {
CollectFieldResult nextLevel = collectFromMergedField(fieldCollectorNormalizedQueryParams, field, mergedField, curLevel + 1);
for (NormalizedField child : nextLevel.children) {
field.addChild(child);
ImmutableList mergedFieldForChild = nextLevel.normalizedFieldToAstFields.get(child);
normalizedFieldToMergedField.put(child, MergedField.newMergedField(mergedFieldForChild).build());
updateFieldToNFMap(child, mergedFieldForChild, fieldNormalizedField);
updateCoordinatedToNFMap(coordinatesToNormalizedFields, child);
buildFieldWithChildren(child,
mergedFieldForChild,
fieldCollectorNormalizedQueryParams,
fieldNormalizedField,
normalizedFieldToMergedField,
coordinatesToNormalizedFields,
curLevel + 1);
}
}
private void updateFieldToNFMap(NormalizedField normalizedField,
ImmutableList mergedField,
ImmutableListMultimap.Builder fieldToNormalizedField) {
for (Field astField : mergedField) {
fieldToNormalizedField.put(astField, normalizedField);
}
}
private void updateCoordinatedToNFMap(ImmutableListMultimap.Builder coordinatesToNormalizedFields, NormalizedField topLevel) {
for (String objectType : topLevel.getObjectTypeNames()) {
FieldCoordinates coordinates = FieldCoordinates.coordinates(objectType, topLevel.getFieldName());
coordinatesToNormalizedFields.put(coordinates, topLevel);
}
}
public static class CollectFieldResult {
private final Collection children;
private final ImmutableListMultimap normalizedFieldToAstFields;
public CollectFieldResult(Collection children, ImmutableListMultimap normalizedFieldToAstFields) {
this.children = children;
this.normalizedFieldToAstFields = normalizedFieldToAstFields;
}
}
public CollectFieldResult collectFromMergedField(FieldCollectorNormalizedQueryParams parameters,
NormalizedField normalizedField,
ImmutableList mergedField,
int level) {
GraphQLUnmodifiedType fieldType = unwrapAll(normalizedField.getType(parameters.getGraphQLSchema()));
// if not composite we don't have any selectionSet because it is a Scalar or enum
if (!(fieldType instanceof GraphQLCompositeType)) {
return new CollectFieldResult(Collections.emptyList(), ImmutableListMultimap.of());
}
Multimap subFields = LinkedHashMultimap.create();
ImmutableListMultimap.Builder mergedFieldByNormalizedField = ImmutableListMultimap.builder();
Set possibleObjects = resolvePossibleObjects((GraphQLCompositeType) fieldType, parameters.getGraphQLSchema());
for (Field field : mergedField) {
if (field.getSelectionSet() == null) {
continue;
}
this.collectFromSelectionSet(parameters,
field.getSelectionSet(),
subFields,
mergedFieldByNormalizedField,
possibleObjects,
level,
normalizedField);
}
return new CollectFieldResult(new LinkedHashSet<>(subFields.values()), mergedFieldByNormalizedField.build());
}
public CollectFieldResult collectFromOperation(FieldCollectorNormalizedQueryParams parameters,
OperationDefinition operationDefinition,
GraphQLObjectType rootType) {
Multimap subFields = LinkedHashMultimap.create();
ImmutableListMultimap.Builder normalizedFieldToAstFields = ImmutableListMultimap.builder();
Set possibleObjects = new LinkedHashSet<>();
possibleObjects.add(rootType);
this.collectFromSelectionSet(parameters, operationDefinition.getSelectionSet(), subFields, normalizedFieldToAstFields, possibleObjects, 1, null);
return new CollectFieldResult(subFields.values(), normalizedFieldToAstFields.build());
}
private void collectFromSelectionSet(FieldCollectorNormalizedQueryParams parameters,
SelectionSet selectionSet,
Multimap result,
ImmutableListMultimap.Builder mergedFieldByNormalizedField,
Set possibleObjects,
int level,
NormalizedField parent) {
for (Selection> selection : selectionSet.getSelections()) {
if (selection instanceof Field) {
collectField(parameters, result, mergedFieldByNormalizedField, (Field) selection, possibleObjects, level, parent);
} else if (selection instanceof InlineFragment) {
collectInlineFragment(parameters, result, mergedFieldByNormalizedField, (InlineFragment) selection, possibleObjects, level, parent);
} else if (selection instanceof FragmentSpread) {
collectFragmentSpread(parameters, result, mergedFieldByNormalizedField, (FragmentSpread) selection, possibleObjects, level, parent);
}
}
}
private void collectFragmentSpread(FieldCollectorNormalizedQueryParams parameters,
Multimap result,
ImmutableListMultimap.Builder mergedFieldByNormalizedField,
FragmentSpread fragmentSpread,
Set possibleObjects,
int level,
NormalizedField parent) {
if (!conditionalNodes.shouldInclude(parameters.getCoercedVariableValues(), fragmentSpread.getDirectives())) {
return;
}
FragmentDefinition fragmentDefinition = assertNotNull(parameters.getFragmentsByName().get(fragmentSpread.getName()));
if (!conditionalNodes.shouldInclude(parameters.getCoercedVariableValues(), fragmentDefinition.getDirectives())) {
return;
}
GraphQLCompositeType newCondition = (GraphQLCompositeType) parameters.getGraphQLSchema().getType(fragmentDefinition.getTypeCondition().getName());
Set newConditions = narrowDownPossibleObjects(possibleObjects, newCondition, parameters.getGraphQLSchema());
collectFromSelectionSet(parameters, fragmentDefinition.getSelectionSet(), result, mergedFieldByNormalizedField, newConditions, level, parent);
}
private void collectInlineFragment(FieldCollectorNormalizedQueryParams parameters,
Multimap result,
ImmutableListMultimap.Builder mergedFieldByNormalizedField,
InlineFragment inlineFragment,
Set possibleObjects,
int level, NormalizedField parent) {
if (!conditionalNodes.shouldInclude(parameters.getCoercedVariableValues(), inlineFragment.getDirectives())) {
return;
}
Set newPossibleObjects = possibleObjects;
if (inlineFragment.getTypeCondition() != null) {
GraphQLCompositeType newCondition = (GraphQLCompositeType) parameters.getGraphQLSchema().getType(inlineFragment.getTypeCondition().getName());
newPossibleObjects = narrowDownPossibleObjects(possibleObjects, newCondition, parameters.getGraphQLSchema());
}
collectFromSelectionSet(parameters, inlineFragment.getSelectionSet(), result, mergedFieldByNormalizedField, newPossibleObjects, level, parent);
}
private void collectField(FieldCollectorNormalizedQueryParams parameters,
Multimap result,
ImmutableListMultimap.Builder normalizedFieldToMergedField,
Field field,
Set objectTypes,
int level,
NormalizedField parent) {
if (!conditionalNodes.shouldInclude(parameters.getCoercedVariableValues(), field.getDirectives())) {
return;
}
// this means there is actually no possible type for this field and we are done
if (objectTypes.size() == 0) {
return;
}
String resultKey = field.getResultKey();
String fieldName = field.getName();
GraphQLFieldDefinition fieldDefinition = Introspection.getFieldDef(parameters.getGraphQLSchema(), objectTypes.iterator().next(), fieldName);
if (result.containsKey(resultKey)) {
Collection existingNFs = result.get(resultKey);
NormalizedField matchingNF = findMatchingNF(parameters.getGraphQLSchema(), existingNFs, fieldDefinition, field.getArguments());
if (matchingNF != null) {
matchingNF.addObjectTypeNames(map(objectTypes, GraphQLObjectType::getName));
normalizedFieldToMergedField.put(matchingNF, field);
return;
}
}
// this means we have no existing NF
Map argumentValues = valuesResolver.getArgumentValues(fieldDefinition.getArguments(), field.getArguments(), parameters.getCoercedVariableValues());
Map normalizedArgumentValues = null;
if (parameters.getNormalizedVariableValues() != null) {
normalizedArgumentValues = valuesResolver.getNormalizedArgumentValues(fieldDefinition.getArguments(), field.getArguments(), parameters.getNormalizedVariableValues());
}
ImmutableList objectTypeNames = map(objectTypes, GraphQLObjectType::getName);
NormalizedField normalizedField = NormalizedField.newNormalizedField()
.alias(field.getAlias())
.resolvedArguments(argumentValues)
.normalizedArguments(normalizedArgumentValues)
.astArguments(field.getArguments())
.objectTypeNames(objectTypeNames)
.fieldName(fieldName)
.level(level)
.parent(parent)
.build();
result.put(resultKey, normalizedField);
normalizedFieldToMergedField.put(normalizedField, field);
}
private NormalizedField findMatchingNF(GraphQLSchema schema, Collection normalizedFields, GraphQLFieldDefinition fieldDefinition, List arguments) {
for (NormalizedField nf : normalizedFields) {
GraphQLFieldDefinition nfFieldDefinition = nf.getOneFieldDefinition(schema);
// same field name
if (!nfFieldDefinition.getName().equals(fieldDefinition.getName())) {
continue;
}
// same type
if (!simplePrint(nfFieldDefinition.getType()).equals(simplePrint(fieldDefinition.getType()))) {
continue;
}
// same arguments
if (!sameArguments(nf.getAstArguments(), arguments)) {
continue;
}
return nf;
}
return null;
}
// copied from graphql.validation.rules.OverlappingFieldsCanBeMerged
private boolean sameArguments(List arguments1, List arguments2) {
if (arguments1.size() != arguments2.size()) {
return false;
}
for (Argument argument : arguments1) {
Argument matchedArgument = findArgumentByName(argument.getName(), arguments2);
if (matchedArgument == null) {
return false;
}
if (!AstComparator.sameValue(argument.getValue(), matchedArgument.getValue())) {
return false;
}
}
return true;
}
private Argument findArgumentByName(String name, List arguments) {
for (Argument argument : arguments) {
if (argument.getName().equals(name)) {
return argument;
}
}
return null;
}
private Set narrowDownPossibleObjects(Set currentOnes,
GraphQLCompositeType typeCondition,
GraphQLSchema graphQLSchema) {
ImmutableSet resolvedTypeCondition = resolvePossibleObjects(typeCondition, graphQLSchema);
if (currentOnes.size() == 0) {
return resolvedTypeCondition;
}
return Sets.intersection(currentOnes, resolvedTypeCondition);
}
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 types = ((GraphQLUnionType) type).getTypes();
return ImmutableSet.copyOf((types));
} else {
return Assert.assertShouldNeverHappen();
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy