Skip to content

Commit c7e90f0

Browse files
committed
Add limit when recursively walking selection set
1 parent 6a10fb5 commit c7e90f0

File tree

6 files changed

+143
-69
lines changed

6 files changed

+143
-69
lines changed

crates/apollo-compiler/src/diagnostics.rs

+2
Original file line numberDiff line numberDiff line change
@@ -244,6 +244,8 @@ pub(crate) enum DiagnosticData {
244244
/// Name of the argument where variable is used
245245
arg_name: String,
246246
},
247+
#[error("too much recursion")]
248+
RecursionError {},
247249
}
248250

249251
impl DiagnosticData {

crates/apollo-compiler/src/validation/fragment.rs

-1
Original file line numberDiff line numberDiff line change
@@ -316,7 +316,6 @@ pub(crate) fn validate_fragment_cycles(
316316
}
317317

318318
if let Some(fragment) = named_fragments.get(&spread.fragment_name) {
319-
println!("going into {}", spread.fragment_name);
320319
detect_fragment_cycles(
321320
named_fragments,
322321
&fragment.selection_set,

crates/apollo-compiler/src/validation/variable.rs

+49-48
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,15 @@
11
use crate::{
22
ast,
33
diagnostics::{ApolloDiagnostic, DiagnosticData, Label},
4-
schema, FileId, Node, NodeLocation, ValidationDatabase,
4+
schema,
5+
validation::{RecursionGuard, RecursionStack},
6+
FileId, Node, NodeLocation, ValidationDatabase,
57
};
68
use std::collections::hash_map::Entry;
79
use std::collections::{HashMap, HashSet};
810

11+
use super::RecursionLimitError;
12+
913
pub(crate) fn validate_variable_definitions2(
1014
db: &dyn ValidationDatabase,
1115
variables: &[Node<ast::VariableDefinition>],
@@ -107,16 +111,11 @@ pub(crate) fn validate_variable_definitions2(
107111
diagnostics
108112
}
109113

110-
enum SelectionPathElement<'ast> {
111-
Field(&'ast ast::Field),
112-
Fragment(&'ast ast::FragmentSpread),
113-
InlineFragment(&'ast ast::InlineFragment),
114-
}
115114
fn walk_selections(
116115
document: &ast::Document,
117116
selections: &[ast::Selection],
118117
mut f: impl FnMut(&ast::Selection),
119-
) {
118+
) -> Result<(), RecursionLimitError> {
120119
type NamedFragments = HashMap<ast::Name, Node<ast::FragmentDefinition>>;
121120
let named_fragments: NamedFragments = document
122121
.definitions
@@ -129,55 +128,48 @@ fn walk_selections(
129128
})
130129
.collect();
131130

132-
fn walk_selections_inner<'ast: 'path, 'path>(
131+
fn walk_selections_inner<'ast, 'guard>(
133132
named_fragments: &'ast NamedFragments,
134133
selections: &'ast [ast::Selection],
135-
path: &mut Vec<SelectionPathElement<'path>>,
134+
guard: &mut RecursionGuard<'guard>,
136135
f: &mut dyn FnMut(&ast::Selection),
137-
) {
136+
) -> Result<(), RecursionLimitError> {
138137
for selection in selections {
139138
f(selection);
140139
match selection {
141140
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)?;
145142
}
146143
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) {
151149
continue;
152150
}
153151

154-
path.push(SelectionPathElement::Fragment(fragment));
155152
if let Some(fragment_definition) = named_fragments.get(&fragment.fragment_name)
156153
{
157154
walk_selections_inner(
158155
named_fragments,
159156
&fragment_definition.selection_set,
160-
path,
157+
&mut guard.push(&fragment.fragment_name)?,
161158
f,
162-
);
159+
)?;
163160
}
164-
path.pop();
165161
}
166162
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)?;
170164
}
171165
}
172166
}
167+
Ok(())
173168
}
174169

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
181173
}
182174

183175
fn variables_in_value(value: &ast::Value) -> impl Iterator<Item = ast::Name> + '_ {
@@ -220,6 +212,8 @@ pub(crate) fn validate_unused_variables(
220212
file_id: FileId,
221213
operation: Node<ast::OperationDefinition>,
222214
) -> Vec<ApolloDiagnostic> {
215+
let mut diagnostics = Vec::new();
216+
223217
let defined_vars: HashSet<_> = operation
224218
.variables
225219
.iter()
@@ -236,24 +230,31 @@ pub(crate) fn validate_unused_variables(
236230
})
237231
.collect();
238232
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+
}
257258

258259
let unused_vars = defined_vars.difference(&used_vars);
259260

crates/apollo-compiler/test_data/diagnostics/0070_self_referential_directive_definition.txt

+27-13
Original file line numberDiff line numberDiff line change
@@ -5,14 +5,14 @@ Error: `invalidExample` directive definition cannot reference itself
55
│ ────────────┬──────────── ───────┬───────
66
│ ╰────────────────────────────────────────── recursive directive definition
77
│ │
8-
│ ╰───────── refers to itself here
8+
│ ╰───────── `invalidExample` circularly references `invalidExample` here
99
───╯
1010
Error: `deprecatedType` directive definition cannot reference itself
1111
╭─[0070_self_referential_directive_definition.graphql:11:1]
1212
1313
10 │ extend scalar String @deprecatedType(reason: "use OurCustomString instead")
1414
│ ───────────────────────────┬──────────────────────────
15-
│ ╰──────────────────────────── refers to itself here
15+
│ ╰──────────────────────────── `deprecatedType` circularly references `deprecatedType` here
1616
11 │ directive @deprecatedType(reason: String!) on OBJECT | INTERFACE | ENUM | SCALAR | UNION
1717
│ ────────────┬────────────
1818
│ ╰────────────── recursive directive definition
@@ -21,39 +21,53 @@ Error: `loopA` directive definition cannot reference itself
2121
╭─[0070_self_referential_directive_definition.graphql:13:1]
2222
2323
13 │ directive @loopA(arg: Boolean @loopB) on ARGUMENT_DEFINITION
24-
│ ────────┬───────
25-
│ ╰───────── recursive directive definition
26-
24+
│ ────────┬─────── ───┬──
25+
│ ╰───────────────────────────── recursive directive definition
26+
│ │
27+
│ ╰──── `loopA` references `loopB` here...
28+
14 │ directive @loopB(arg: Boolean @loopC) on ARGUMENT_DEFINITION
29+
│ ───┬──
30+
│ ╰──── `loopB` references `loopC` here...
2731
15 │ directive @loopC(arg: Boolean @loopA) on ARGUMENT_DEFINITION
2832
│ ───┬──
29-
│ ╰──── refers to itself here
33+
│ ╰──── `loopC` circularly references `loopA` here
3034
────╯
3135
Error: `loopB` directive definition cannot reference itself
3236
╭─[0070_self_referential_directive_definition.graphql:14:1]
3337
3438
13 │ directive @loopA(arg: Boolean @loopB) on ARGUMENT_DEFINITION
3539
│ ───┬──
36-
│ ╰──── refers to itself here
40+
│ ╰──── `loopA` circularly references `loopB` here
3741
14 │ directive @loopB(arg: Boolean @loopC) on ARGUMENT_DEFINITION
38-
│ ────────┬───────
39-
│ ╰───────── recursive directive definition
42+
│ ────────┬─────── ───┬──
43+
│ ╰───────────────────────────── recursive directive definition
44+
│ │
45+
│ ╰──── `loopB` references `loopC` here...
46+
15 │ directive @loopC(arg: Boolean @loopA) on ARGUMENT_DEFINITION
47+
│ ───┬──
48+
│ ╰──── `loopC` references `loopA` here...
4049
────╯
4150
Error: `loopC` directive definition cannot reference itself
4251
╭─[0070_self_referential_directive_definition.graphql:15:1]
4352
53+
13 │ directive @loopA(arg: Boolean @loopB) on ARGUMENT_DEFINITION
54+
│ ───┬──
55+
│ ╰──── `loopA` references `loopB` here...
4456
14 │ directive @loopB(arg: Boolean @loopC) on ARGUMENT_DEFINITION
4557
│ ───┬──
46-
│ ╰──── refers to itself here
58+
│ ╰──── `loopB` circularly references `loopC` here
4759
15 │ directive @loopC(arg: Boolean @loopA) on ARGUMENT_DEFINITION
48-
│ ────────┬───────
49-
│ ╰───────── recursive directive definition
60+
│ ────────┬─────── ───┬──
61+
│ ╰───────────────────────────── recursive directive definition
62+
│ │
63+
│ ╰──── `loopC` references `loopA` here...
5064
────╯
5165
Error: `wrong` directive definition cannot reference itself
5266
╭─[0070_self_referential_directive_definition.graphql:21:1]
5367
5468
18 │ extend enum Enum { RECURSIVE @wrong }
5569
│ ───┬──
56-
│ ╰──── refers to itself here
70+
│ ╰──── `wrong` circularly references `wrong` here
5771
5872
21 │ directive @wrong(input: InputObject) on INPUT_FIELD_DEFINITION | ENUM_VALUE
5973
│ ────────┬───────

crates/apollo-compiler/test_data/diagnostics/0084_circular_non_nullable_input_objects.txt

+48-5
Original file line numberDiff line numberDiff line change
@@ -2,50 +2,93 @@ Error: `First` input object cannot reference itself
22
╭─[0084_circular_non_nullable_input_objects.graphql:6:1]
33
44
6 │ ╭─▶ input First {
5+
7 │ │ second: Second!
6+
│ │ ───────┬───────
7+
│ │ ╰───────── `First` references `second` here...
58
┆ ┆
69
9 │ ├─▶ }
710
│ │
811
│ ╰─────── cyclical input object definition
912
13+
12 │ third: Third!
14+
│ ──────┬──────
15+
│ ╰──────── `second` references `third` here...
16+
17+
17 │ fourth: Fourth!
18+
│ ───────┬───────
19+
│ ╰───────── `third` references `fourth` here...
20+
1021
22 │ first: First!
1122
│ ──────┬──────
12-
│ ╰──────── refers to itself here
23+
│ ╰──────── `fourth` circularly references `first` here
1324
────╯
1425
Error: `Second` input object cannot reference itself
1526
╭─[0084_circular_non_nullable_input_objects.graphql:11:1]
1627
1728
7 │ second: Second!
1829
│ ───────┬───────
19-
│ ╰───────── refers to itself here
30+
│ ╰───────── `first` circularly references `second` here
2031
2132
11 │ ╭─▶ input Second {
33+
12 │ │ third: Third!
34+
│ │ ──────┬──────
35+
│ │ ╰──────── `Second` references `third` here...
2236
┆ ┆
2337
14 │ ├─▶ }
2438
│ │
2539
│ ╰─────── cyclical input object definition
40+
41+
17 │ fourth: Fourth!
42+
│ ───────┬───────
43+
│ ╰───────── `third` references `fourth` here...
44+
45+
22 │ first: First!
46+
│ ──────┬──────
47+
│ ╰──────── `fourth` references `first` here...
2648
────╯
2749
Error: `Third` input object cannot reference itself
2850
╭─[0084_circular_non_nullable_input_objects.graphql:16:1]
2951
52+
7 │ second: Second!
53+
│ ───────┬───────
54+
│ ╰───────── `first` references `second` here...
55+
3056
12 │ third: Third!
3157
│ ──────┬──────
32-
│ ╰──────── refers to itself here
58+
│ ╰──────── `second` circularly references `third` here
3359
3460
16 │ ╭─▶ input Third {
61+
17 │ │ fourth: Fourth!
62+
│ │ ───────┬───────
63+
│ │ ╰───────── `Third` references `fourth` here...
3564
┆ ┆
3665
19 │ ├─▶ }
3766
│ │
3867
│ ╰─────── cyclical input object definition
68+
69+
22 │ first: First!
70+
│ ──────┬──────
71+
│ ╰──────── `fourth` references `first` here...
3972
────╯
4073
Error: `Fourth` input object cannot reference itself
4174
╭─[0084_circular_non_nullable_input_objects.graphql:21:1]
4275
76+
7 │ second: Second!
77+
│ ───────┬───────
78+
│ ╰───────── `first` references `second` here...
79+
80+
12 │ third: Third!
81+
│ ──────┬──────
82+
│ ╰──────── `second` references `third` here...
83+
4384
17 │ fourth: Fourth!
4485
│ ───────┬───────
45-
│ ╰───────── refers to itself here
86+
│ ╰───────── `third` circularly references `fourth` here
4687
4788
21 │ ╭─▶ input Fourth {
48-
┆ ┆
89+
22 │ │ first: First!
90+
│ │ ──────┬──────
91+
│ │ ╰──────── `Fourth` references `first` here...
4992
23 │ ├─▶ }
5093
│ │
5194
│ ╰────── cyclical input object definition

crates/apollo-compiler/tests/validation/fragments.rs

+17-2
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,7 @@ fn long_fragment_chains_do_not_overflow_stack() {
1313
"#
1414
.to_string();
1515

16-
let fragments: usize = 10_000;
16+
let fragments: usize = 1_000;
1717
for i in 1..fragments {
1818
query.push_str(&format!(
1919
"
@@ -41,5 +41,20 @@ fn long_fragment_chains_do_not_overflow_stack() {
4141
),
4242
"overflow.graphql",
4343
);
44-
executable.validate(&schema).unwrap();
44+
45+
let errors = executable
46+
.validate(&schema)
47+
.expect_err("must have recursion errors");
48+
49+
let expected = expect_test::expect![[r#"
50+
Error: too much recursion
51+
Error: `typeFragment1` fragment cannot reference itself
52+
╭─[overflow.graphql:11:11]
53+
54+
11 │ fragment typeFragment1 on __Type {
55+
│ ───────────┬──────────
56+
│ ╰──────────── recursive fragment definition
57+
────╯
58+
"#]];
59+
expected.assert_eq(&errors.to_string_no_color());
4560
}

0 commit comments

Comments
 (0)