graphql.validation.rules.PossibleFragmentSpreads Maven / Gradle / Ivy
package graphql.validation.rules;
import graphql.Assert;
import graphql.execution.TypeFromAST;
import graphql.language.FragmentDefinition;
import graphql.language.FragmentSpread;
import graphql.language.InlineFragment;
import graphql.schema.GraphQLCompositeType;
import graphql.schema.GraphQLInterfaceType;
import graphql.schema.GraphQLObjectType;
import graphql.schema.GraphQLOutputType;
import graphql.schema.GraphQLType;
import graphql.schema.GraphQLUnionType;
import graphql.schema.SchemaUtil;
import graphql.validation.AbstractRule;
import graphql.validation.ValidationContext;
import graphql.validation.ValidationError;
import graphql.validation.ValidationErrorCollector;
import graphql.validation.ValidationErrorType;
import java.util.Collections;
import java.util.List;
public class PossibleFragmentSpreads extends AbstractRule {
public PossibleFragmentSpreads(ValidationContext validationContext, ValidationErrorCollector validationErrorCollector) {
super(validationContext, validationErrorCollector);
}
@Override
public void checkInlineFragment(InlineFragment inlineFragment) {
GraphQLOutputType fragType = getValidationContext().getOutputType();
GraphQLCompositeType parentType = getValidationContext().getParentType();
if (fragType == null || parentType == null) return;
if (!doTypesOverlap(fragType, parentType)) {
String message = String.format("Fragment cannot be spread here as objects of " +
"type %s can never be of type %s", parentType, fragType);
addError(new ValidationError(ValidationErrorType.InvalidFragmentType, inlineFragment.getSourceLocation(), message));
}
}
@Override
public void checkFragmentSpread(FragmentSpread fragmentSpread) {
FragmentDefinition fragment = getValidationContext().getFragment(fragmentSpread.getName());
if (fragment == null) return;
GraphQLType typeCondition = TypeFromAST.getTypeFromAST(getValidationContext().getSchema(), fragment.getTypeCondition());
GraphQLCompositeType parentType = getValidationContext().getParentType();
if (typeCondition == null || parentType == null) return;
if (!doTypesOverlap(typeCondition, parentType)) {
String message = String.format("Fragment %s cannot be spread here as objects of " +
"type %s can never be of type %s", fragmentSpread.getName(), parentType, typeCondition);
addError(new ValidationError(ValidationErrorType.InvalidFragmentType, fragmentSpread.getSourceLocation(), message));
}
}
private boolean doTypesOverlap(GraphQLType type, GraphQLCompositeType parent) {
if (type == parent) {
return true;
}
List extends GraphQLType> possibleParentTypes = getPossibleType(parent);
List extends GraphQLType> possibleConditionTypes = getPossibleType(type);
return !Collections.disjoint(possibleParentTypes, possibleConditionTypes);
}
private List extends GraphQLType> getPossibleType(GraphQLType type) {
List extends GraphQLType> possibleConditionTypes = null;
if (type instanceof GraphQLObjectType) {
possibleConditionTypes = Collections.singletonList(type);
} else if (type instanceof GraphQLInterfaceType) {
possibleConditionTypes = new SchemaUtil().findImplementations(getValidationContext().getSchema(), (GraphQLInterfaceType) type);
} else if (type instanceof GraphQLUnionType) {
possibleConditionTypes = ((GraphQLUnionType) type).getTypes();
} else {
Assert.assertShouldNeverHappen();
}
return possibleConditionTypes;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy