1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
|
package validator
import (
"github.com/vektah/gqlparser/ast"
. "github.com/vektah/gqlparser/validator"
)
func init() {
AddRule("PossibleFragmentSpreads", func(observers *Events, addError AddErrFunc) {
validate := func(walker *Walker, parentDef *ast.Definition, fragmentName string, emitError func()) {
if parentDef == nil {
return
}
var parentDefs []*ast.Definition
switch parentDef.Kind {
case ast.Object:
parentDefs = []*ast.Definition{parentDef}
case ast.Interface, ast.Union:
parentDefs = walker.Schema.GetPossibleTypes(parentDef)
default:
panic("unexpected type")
}
fragmentDefType := walker.Schema.Types[fragmentName]
if fragmentDefType == nil {
return
}
if !fragmentDefType.IsCompositeType() {
// checked by FragmentsOnCompositeTypes
return
}
fragmentDefs := walker.Schema.GetPossibleTypes(fragmentDefType)
for _, fragmentDef := range fragmentDefs {
for _, parentDef := range parentDefs {
if parentDef.Name == fragmentDef.Name {
return
}
}
}
emitError()
}
observers.OnInlineFragment(func(walker *Walker, inlineFragment *ast.InlineFragment) {
validate(walker, inlineFragment.ObjectDefinition, inlineFragment.TypeCondition, func() {
addError(
Message(`Fragment cannot be spread here as objects of type "%s" can never be of type "%s".`, inlineFragment.ObjectDefinition.Name, inlineFragment.TypeCondition),
At(inlineFragment.Position),
)
})
})
observers.OnFragmentSpread(func(walker *Walker, fragmentSpread *ast.FragmentSpread) {
if fragmentSpread.Definition == nil {
return
}
validate(walker, fragmentSpread.ObjectDefinition, fragmentSpread.Definition.TypeCondition, func() {
addError(
Message(`Fragment "%s" cannot be spread here as objects of type "%s" can never be of type "%s".`, fragmentSpread.Name, fragmentSpread.ObjectDefinition.Name, fragmentSpread.Definition.TypeCondition),
At(fragmentSpread.Position),
)
})
})
})
}
|