import { inspect } from '../../jsutils/inspect';
import type { Maybe } from '../../jsutils/Maybe';
import { GraphQLError } from '../../error/GraphQLError';
import type { ASTVisitor } from '../../language/visitor';
import type { GraphQLCompositeType } from '../../type/definition';
import { isCompositeType } from '../../type/definition';
import { doTypesOverlap } from '../../utilities/typeComparators';
import { typeFromAST } from '../../utilities/typeFromAST';
import type { ValidationContext } from '../ValidationContext';
export function PossibleFragmentSpreadsRule(
context: ValidationContext,
): ASTVisitor {
return {
InlineFragment(node) {
const fragType = context.getType();
const parentType = context.getParentType();
if (
isCompositeType(fragType) &&
isCompositeType(parentType) &&
!doTypesOverlap(context.getSchema(), fragType, parentType)
) {
const parentTypeStr = inspect(parentType);
const fragTypeStr = inspect(fragType);
context.reportError(
new GraphQLError(
`Fragment cannot be spread here as objects of type "${parentTypeStr}" can never be of type "${fragTypeStr}".`,
{ nodes: node },
),
);
}
},
FragmentSpread(node) {
const fragName = node.name.value;
const fragType = getFragmentType(context, fragName);
const parentType = context.getParentType();
if (
fragType &&
parentType &&
!doTypesOverlap(context.getSchema(), fragType, parentType)
) {
const parentTypeStr = inspect(parentType);
const fragTypeStr = inspect(fragType);
context.reportError(
new GraphQLError(
`Fragment "${fragName}" cannot be spread here as objects of type "${parentTypeStr}" can never be of type "${fragTypeStr}".`,
{ nodes: node },
),
);
}
},
};
}
function getFragmentType(
context: ValidationContext,
name: string,
): Maybe<GraphQLCompositeType> {
const frag = context.getFragment(name);
if (frag) {
const type = typeFromAST(context.getSchema(), frag.typeCondition);
if (isCompositeType(type)) {
return type;
}
}
}