graphql.validation.rules.OverlappingFieldsCanBeMerged Maven / Gradle / Ivy
package graphql.validation.rules;
import com.google.common.collect.ImmutableList;
import graphql.Internal;
import graphql.execution.TypeFromAST;
import graphql.language.Argument;
import graphql.language.AstComparator;
import graphql.language.Field;
import graphql.language.FragmentDefinition;
import graphql.language.FragmentSpread;
import graphql.language.InlineFragment;
import graphql.language.Selection;
import graphql.language.SelectionSet;
import graphql.schema.GraphQLFieldDefinition;
import graphql.schema.GraphQLFieldsContainer;
import graphql.schema.GraphQLInterfaceType;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLOutputType;
import graphql.schema.GraphQLType;
import graphql.schema.GraphQLUnionType;
import graphql.schema.GraphQLUnmodifiedType;
import graphql.validation.AbstractRule;
import graphql.validation.ValidationContext;
import graphql.validation.ValidationErrorCollector;
import java.util.ArrayList;
import java.util.Collections;
import java.util.LinkedHashMap;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Map;
import java.util.Objects;
import java.util.Set;
import static graphql.collect.ImmutableKit.addToList;
import static graphql.collect.ImmutableKit.emptyList;
import static graphql.schema.GraphQLTypeUtil.isEnum;
import static graphql.schema.GraphQLTypeUtil.isList;
import static graphql.schema.GraphQLTypeUtil.isNonNull;
import static graphql.schema.GraphQLTypeUtil.isNotWrapped;
import static graphql.schema.GraphQLTypeUtil.isNullable;
import static graphql.schema.GraphQLTypeUtil.isScalar;
import static graphql.schema.GraphQLTypeUtil.simplePrint;
import static graphql.schema.GraphQLTypeUtil.unwrapAll;
import static graphql.schema.GraphQLTypeUtil.unwrapOne;
import static graphql.util.FpKit.filterSet;
import static graphql.util.FpKit.groupingBy;
import static graphql.validation.ValidationErrorType.FieldsConflict;
import static java.lang.String.format;
@Internal
public class OverlappingFieldsCanBeMerged extends AbstractRule {
private final Set> sameResponseShapeChecked = new LinkedHashSet<>();
private final Set> sameForCommonParentsChecked = new LinkedHashSet<>();
private final Set> conflictsReported = new LinkedHashSet<>();
public OverlappingFieldsCanBeMerged(ValidationContext validationContext, ValidationErrorCollector validationErrorCollector) {
super(validationContext, validationErrorCollector);
}
@Override
public void leaveSelectionSet(SelectionSet selectionSet) {
Map> fieldMap = new LinkedHashMap<>();
Set visitedFragmentSpreads = new LinkedHashSet<>();
collectFields(fieldMap, selectionSet, getValidationContext().getOutputType(), visitedFragmentSpreads);
List conflicts = findConflicts(fieldMap);
for (Conflict conflict : conflicts) {
if (conflictsReported.contains(conflict.fields)) {
continue;
}
conflictsReported.add(conflict.fields);
// each error contains a reference to the current querypath via validationContext.getQueryPath()
// queryPath is null for the first selection set
addError(FieldsConflict, conflict.fields, conflict.reason);
}
}
private void collectFields(Map> fieldMap, SelectionSet selectionSet, GraphQLType parentType, Set visitedFragmentSpreads) {
for (Selection selection : selectionSet.getSelections()) {
if (selection instanceof Field) {
collectFieldsForField(fieldMap, parentType, (Field) selection);
} else if (selection instanceof InlineFragment) {
collectFieldsForInlineFragment(fieldMap, visitedFragmentSpreads, parentType, (InlineFragment) selection);
} else if (selection instanceof FragmentSpread) {
collectFieldsForFragmentSpread(fieldMap, visitedFragmentSpreads, (FragmentSpread) selection);
}
}
}
private void collectFieldsForFragmentSpread(Map> fieldMap, Set visitedFragmentSpreads, FragmentSpread fragmentSpread) {
FragmentDefinition fragment = getValidationContext().getFragment(fragmentSpread.getName());
if (fragment == null) {
return;
}
if (visitedFragmentSpreads.contains(fragment.getName())) {
return;
}
visitedFragmentSpreads.add(fragment.getName());
GraphQLType graphQLType = getGraphQLTypeForFragmentDefinition(fragment);
collectFields(fieldMap, fragment.getSelectionSet(), graphQLType, visitedFragmentSpreads);
}
private GraphQLType getGraphQLTypeForFragmentDefinition(FragmentDefinition fragment) {
return TypeFromAST.getTypeFromAST(getValidationContext().getSchema(),
fragment.getTypeCondition());
}
private void collectFieldsForInlineFragment(Map> fieldMap, Set visitedFragmentSpreads, GraphQLType parentType, InlineFragment inlineFragment) {
GraphQLType graphQLType = getGraphQLTypeForInlineFragment(parentType, inlineFragment);
collectFields(fieldMap, inlineFragment.getSelectionSet(), graphQLType, visitedFragmentSpreads);
}
private GraphQLType getGraphQLTypeForInlineFragment(GraphQLType parentType, InlineFragment inlineFragment) {
if (inlineFragment.getTypeCondition() == null) {
return parentType;
}
return TypeFromAST.getTypeFromAST(getValidationContext().getSchema(), inlineFragment.getTypeCondition());
}
private void collectFieldsForField(Map> fieldMap, GraphQLType parentType, Field field) {
String responseName = field.getResultKey();
if (!fieldMap.containsKey(responseName)) {
fieldMap.put(responseName, new LinkedHashSet<>());
}
GraphQLOutputType fieldType = null;
GraphQLUnmodifiedType unwrappedParent = unwrapAll(parentType);
if (unwrappedParent instanceof GraphQLFieldsContainer) {
GraphQLFieldsContainer fieldsContainer = (GraphQLFieldsContainer) unwrappedParent;
GraphQLFieldDefinition fieldDefinition = getVisibleFieldDefinition(fieldsContainer, field);
fieldType = fieldDefinition != null ? fieldDefinition.getType() : null;
}
fieldMap.get(responseName).add(new FieldAndType(field, fieldType, parentType));
}
private GraphQLFieldDefinition getVisibleFieldDefinition(GraphQLFieldsContainer fieldsContainer, Field field) {
return getValidationContext().getSchema().getCodeRegistry().getFieldVisibility().getFieldDefinition(fieldsContainer, field.getName());
}
private List findConflicts(Map> fieldMap) {
/*
* The algorithm implemented here is not the one from the Spec, but is based on
* https://tech.xing.com/graphql-overlapping-fields-can-be-merged-fast-ea6e92e0a01
* . It is not the final version (Listing 11), but Listing 10 adopted to this code base.
*/
List result = new ArrayList<>();
sameResponseShapeByName(fieldMap, emptyList(), result);
sameForCommonParentsByName(fieldMap, emptyList(), result);
return result;
}
private void sameResponseShapeByName(Map> fieldMap, ImmutableList currentPath, List conflictsResult) {
for (Map.Entry> entry : fieldMap.entrySet()) {
if (sameResponseShapeChecked.contains(entry.getValue())) {
continue;
}
ImmutableList newPath = addToList(currentPath, entry.getKey());
sameResponseShapeChecked.add(entry.getValue());
Conflict conflict = requireSameOutputTypeShape(newPath, entry.getValue());
if (conflict != null) {
conflictsResult.add(conflict);
continue;
}
Map> subSelections = mergeSubSelections(entry.getValue());
sameResponseShapeByName(subSelections, newPath, conflictsResult);
}
}
private Map> mergeSubSelections(Set sameNameFields) {
Map> fieldMap = new LinkedHashMap<>();
for (FieldAndType fieldAndType : sameNameFields) {
if (fieldAndType.field.getSelectionSet() != null) {
Set visitedFragmentSpreads = new LinkedHashSet<>();
collectFields(fieldMap, fieldAndType.field.getSelectionSet(), fieldAndType.graphQLType, visitedFragmentSpreads);
}
}
return fieldMap;
}
private void sameForCommonParentsByName(Map> fieldMap, ImmutableList currentPath, List conflictsResult) {
for (Map.Entry> entry : fieldMap.entrySet()) {
List> groups = groupByCommonParents(entry.getValue());
ImmutableList newPath = addToList(currentPath, entry.getKey());
for (Set group : groups) {
if (sameForCommonParentsChecked.contains(group)) {
continue;
}
sameForCommonParentsChecked.add(group);
Conflict conflict = requireSameNameAndArguments(newPath, group);
if (conflict != null) {
conflictsResult.add(conflict);
continue;
}
Map> subSelections = mergeSubSelections(group);
sameForCommonParentsByName(subSelections, newPath, conflictsResult);
}
}
}
private List> groupByCommonParents(Set fields) {
Set abstractTypes = filterSet(fields, fieldAndType -> isInterfaceOrUnion(fieldAndType.parentType));
Set concreteTypes = filterSet(fields, fieldAndType -> fieldAndType.parentType instanceof GraphQLObjectType);
if (concreteTypes.isEmpty()) {
return Collections.singletonList(abstractTypes);
}
Map> groupsByConcreteParent = groupingBy(concreteTypes, fieldAndType -> fieldAndType.parentType);
List> result = new ArrayList<>();
for (ImmutableList concreteGroup : groupsByConcreteParent.values()) {
Set oneResultGroup = new LinkedHashSet<>(concreteGroup);
oneResultGroup.addAll(abstractTypes);
result.add(oneResultGroup);
}
return result;
}
private boolean isInterfaceOrUnion(GraphQLType type) {
return type instanceof GraphQLInterfaceType || type instanceof GraphQLUnionType;
}
private Conflict requireSameNameAndArguments(ImmutableList path, Set fieldAndTypes) {
if (fieldAndTypes.size() <= 1) {
return null;
}
String name = null;
List arguments = null;
List fields = new ArrayList<>();
for (FieldAndType fieldAndType : fieldAndTypes) {
Field field = fieldAndType.field;
fields.add(field);
if (name == null) {
name = field.getName();
arguments = field.getArguments();
continue;
}
if (!field.getName().equals(name)) {
String reason = format("%s: %s and %s are different fields", pathToString(path), name, field.getName());
return new Conflict(reason, fields);
}
if (!sameArguments(field.getArguments(), arguments)) {
String reason = format("%s: they have differing arguments", pathToString(path));
return new Conflict(reason, fields);
}
}
return null;
}
private String pathToString(ImmutableList path) {
return String.join("/", path);
}
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 Conflict requireSameOutputTypeShape(ImmutableList path, Set fieldAndTypes) {
if (fieldAndTypes.size() <= 1) {
return null;
}
List fields = new ArrayList<>();
GraphQLType typeAOriginal = null;
for (FieldAndType fieldAndType : fieldAndTypes) {
fields.add(fieldAndType.field);
if (typeAOriginal == null) {
typeAOriginal = fieldAndType.graphQLType;
continue;
}
GraphQLType typeA = typeAOriginal;
GraphQLType typeB = fieldAndType.graphQLType;
while (true) {
if (isNonNull(typeA) || isNonNull(typeB)) {
if (isNullable(typeA) || isNullable(typeB)) {
String reason = format("%s: fields have different nullability shapes", pathToString(path));
return new Conflict(reason, fields);
}
}
if (isList(typeA) || isList(typeB)) {
if (!isList(typeA) || !isList(typeB)) {
String reason = format("%s: fields have different list shapes", pathToString(path));
return new Conflict(reason, fields);
}
}
if (isNotWrapped(typeA) && isNotWrapped(typeB)) {
break;
}
typeA = unwrapOne(typeA);
typeB = unwrapOne(typeB);
}
if (isScalar(typeA) || isScalar(typeB)) {
if (!sameType(typeA, typeB)) {
return mkNotSameTypeError(path, fields, typeA, typeB);
}
}
if (isEnum(typeA) || isEnum(typeB)) {
if (!sameType(typeA, typeB)) {
return mkNotSameTypeError(path, fields, typeA, typeB);
}
}
}
return null;
}
private Conflict mkNotSameTypeError(ImmutableList path, List fields, GraphQLType typeA, GraphQLType typeB) {
String name1 = typeA != null ? simplePrint(typeA) : "null";
String name2 = typeB != null ? simplePrint(typeB) : "null";
String reason = format("%s: they return differing types %s and %s", pathToString(path), name1, name2);
return new Conflict(reason, fields);
}
private boolean sameType(GraphQLType type1, GraphQLType type2) {
if (type1 == null || type2 == null) {
return true;
}
return type1.equals(type2);
}
private static class FieldAndType {
final Field field;
final GraphQLType graphQLType;
final GraphQLType parentType;
public FieldAndType(Field field, GraphQLType graphQLType, GraphQLType parentType) {
this.field = field;
this.graphQLType = graphQLType;
this.parentType = parentType;
}
@Override
public String toString() {
return "FieldAndType{" +
"field=" + field +
", graphQLType=" + graphQLType +
", parentType=" + parentType +
'}';
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
FieldAndType that = (FieldAndType) o;
return Objects.equals(field, that.field);
}
@Override
public int hashCode() {
return Objects.hashCode(field);
}
}
private static class Conflict {
final String reason;
final Set fields = new LinkedHashSet<>();
public Conflict(String reason, List fields) {
this.reason = reason;
this.fields.addAll(fields);
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy