1
1
use crate :: {
2
2
ast,
3
3
diagnostics:: { ApolloDiagnostic , DiagnosticData , Label } ,
4
- schema, FileId , Node , NodeLocation , ValidationDatabase ,
4
+ schema,
5
+ validation:: { RecursionGuard , RecursionStack } ,
6
+ FileId , Node , NodeLocation , ValidationDatabase ,
5
7
} ;
6
8
use std:: collections:: hash_map:: Entry ;
7
9
use std:: collections:: { HashMap , HashSet } ;
8
10
11
+ use super :: RecursionLimitError ;
12
+
9
13
pub ( crate ) fn validate_variable_definitions2 (
10
14
db : & dyn ValidationDatabase ,
11
15
variables : & [ Node < ast:: VariableDefinition > ] ,
@@ -107,16 +111,11 @@ pub(crate) fn validate_variable_definitions2(
107
111
diagnostics
108
112
}
109
113
110
- enum SelectionPathElement < ' ast > {
111
- Field ( & ' ast ast:: Field ) ,
112
- Fragment ( & ' ast ast:: FragmentSpread ) ,
113
- InlineFragment ( & ' ast ast:: InlineFragment ) ,
114
- }
115
114
fn walk_selections (
116
115
document : & ast:: Document ,
117
116
selections : & [ ast:: Selection ] ,
118
117
mut f : impl FnMut ( & ast:: Selection ) ,
119
- ) {
118
+ ) -> Result < ( ) , RecursionLimitError > {
120
119
type NamedFragments = HashMap < ast:: Name , Node < ast:: FragmentDefinition > > ;
121
120
let named_fragments: NamedFragments = document
122
121
. definitions
@@ -129,55 +128,48 @@ fn walk_selections(
129
128
} )
130
129
. collect ( ) ;
131
130
132
- fn walk_selections_inner < ' ast : ' path , ' path > (
131
+ fn walk_selections_inner < ' ast , ' guard > (
133
132
named_fragments : & ' ast NamedFragments ,
134
133
selections : & ' ast [ ast:: Selection ] ,
135
- path : & mut Vec < SelectionPathElement < ' path > > ,
134
+ guard : & mut RecursionGuard < ' guard > ,
136
135
f : & mut dyn FnMut ( & ast:: Selection ) ,
137
- ) {
136
+ ) -> Result < ( ) , RecursionLimitError > {
138
137
for selection in selections {
139
138
f ( selection) ;
140
139
match selection {
141
140
ast:: Selection :: Field ( field) => {
142
- path. push ( SelectionPathElement :: Field ( field) ) ;
143
- walk_selections_inner ( named_fragments, & field. selection_set , path, f) ;
144
- path. pop ( ) ;
141
+ walk_selections_inner ( named_fragments, & field. selection_set , guard, f) ?;
145
142
}
146
143
ast:: Selection :: FragmentSpread ( fragment) => {
147
- // prevent recursion
148
- if path. iter ( ) . any ( |element| {
149
- matches ! ( element, SelectionPathElement :: Fragment ( walked) if walked. fragment_name == fragment. fragment_name)
150
- } ) {
144
+ // Prevent chasing a cyclical reference.
145
+ // Note we do not report `CycleError::Recursed` here, as that is already caught
146
+ // by the cyclical fragment validation--we just need to ensure that we don't
147
+ // overflow the stack.
148
+ if guard. contains ( & fragment. fragment_name ) {
151
149
continue ;
152
150
}
153
151
154
- path. push ( SelectionPathElement :: Fragment ( fragment) ) ;
155
152
if let Some ( fragment_definition) = named_fragments. get ( & fragment. fragment_name )
156
153
{
157
154
walk_selections_inner (
158
155
named_fragments,
159
156
& fragment_definition. selection_set ,
160
- path ,
157
+ & mut guard . push ( & fragment . fragment_name ) ? ,
161
158
f,
162
- ) ;
159
+ ) ? ;
163
160
}
164
- path. pop ( ) ;
165
161
}
166
162
ast:: Selection :: InlineFragment ( fragment) => {
167
- path. push ( SelectionPathElement :: InlineFragment ( fragment) ) ;
168
- walk_selections_inner ( named_fragments, & fragment. selection_set , path, f) ;
169
- path. pop ( ) ;
163
+ walk_selections_inner ( named_fragments, & fragment. selection_set , guard, f) ?;
170
164
}
171
165
}
172
166
}
167
+ Ok ( ( ) )
173
168
}
174
169
175
- walk_selections_inner (
176
- & named_fragments,
177
- selections,
178
- & mut Default :: default ( ) ,
179
- & mut f,
180
- ) ;
170
+ let mut stack = RecursionStack :: new ( 500 ) ;
171
+ let result = walk_selections_inner ( & named_fragments, selections, & mut stack. guard ( ) , & mut f) ;
172
+ result
181
173
}
182
174
183
175
fn variables_in_value ( value : & ast:: Value ) -> impl Iterator < Item = ast:: Name > + ' _ {
@@ -220,6 +212,8 @@ pub(crate) fn validate_unused_variables(
220
212
file_id : FileId ,
221
213
operation : Node < ast:: OperationDefinition > ,
222
214
) -> Vec < ApolloDiagnostic > {
215
+ let mut diagnostics = Vec :: new ( ) ;
216
+
223
217
let defined_vars: HashSet < _ > = operation
224
218
. variables
225
219
. iter ( )
@@ -236,24 +230,31 @@ pub(crate) fn validate_unused_variables(
236
230
} )
237
231
. collect ( ) ;
238
232
let mut used_vars = HashSet :: < ast:: Name > :: new ( ) ;
239
- walk_selections (
240
- & db. ast ( file_id) ,
241
- & operation. selection_set ,
242
- |selection| match selection {
243
- ast:: Selection :: Field ( field) => {
244
- used_vars. extend ( variables_in_directives ( & field. directives ) ) ;
245
- used_vars. extend ( variables_in_arguments ( & field. arguments ) ) ;
246
- }
247
- ast:: Selection :: FragmentSpread ( fragment) => {
248
- used_vars. extend ( variables_in_directives ( & fragment. directives ) ) ;
249
- }
250
- ast:: Selection :: InlineFragment ( fragment) => {
251
- used_vars. extend ( variables_in_directives ( & fragment. directives ) ) ;
252
- }
253
- } ,
254
- ) ;
255
-
256
- let mut diagnostics = Vec :: new ( ) ;
233
+ let walked =
234
+ walk_selections (
235
+ & db. ast ( file_id) ,
236
+ & operation. selection_set ,
237
+ |selection| match selection {
238
+ ast:: Selection :: Field ( field) => {
239
+ used_vars. extend ( variables_in_directives ( & field. directives ) ) ;
240
+ used_vars. extend ( variables_in_arguments ( & field. arguments ) ) ;
241
+ }
242
+ ast:: Selection :: FragmentSpread ( fragment) => {
243
+ used_vars. extend ( variables_in_directives ( & fragment. directives ) ) ;
244
+ }
245
+ ast:: Selection :: InlineFragment ( fragment) => {
246
+ used_vars. extend ( variables_in_directives ( & fragment. directives ) ) ;
247
+ }
248
+ } ,
249
+ ) ;
250
+ if walked. is_err ( ) {
251
+ diagnostics. push ( ApolloDiagnostic :: new (
252
+ db,
253
+ None ,
254
+ DiagnosticData :: RecursionError { } ,
255
+ ) ) ;
256
+ return diagnostics;
257
+ }
257
258
258
259
let unused_vars = defined_vars. difference ( & used_vars) ;
259
260
0 commit comments