Adjust for-loop's pattern matching rules (#3308)

This commit is contained in:
Leedehai 2024-01-31 04:12:06 -05:00 committed by GitHub
parent ce5abf5a4e
commit 51854ba4df
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
2 changed files with 64 additions and 14 deletions

View File

@ -108,11 +108,11 @@ impl Eval for ast::ForLoop<'_> {
let mut output = Value::None; let mut output = Value::None;
macro_rules! iter { macro_rules! iter {
(for $pat:ident in $iter:expr) => {{ (for $pat:ident in $iterable:expr) => {{
vm.scopes.enter(); vm.scopes.enter();
#[allow(unused_parens)] #[allow(unused_parens)]
for value in $iter { for value in $iterable {
destructure(vm, $pat, value.into_value())?; destructure(vm, $pat, value.into_value())?;
let body = self.body(); let body = self.body();
@ -134,29 +134,30 @@ impl Eval for ast::ForLoop<'_> {
}}; }};
} }
let pattern = self.pattern();
let iterable = self.iterable().eval(vm)?; let iterable = self.iterable().eval(vm)?;
let iterable_type = iterable.ty(); let iterable_type = iterable.ty();
let pattern = self.pattern();
match (&pattern, iterable) { use ast::Pattern;
(ast::Pattern::Normal(_), Value::Str(string)) => { match (pattern, iterable) {
// Iterate over graphemes of string. (_, Value::Array(array)) => {
iter!(for pattern in string.as_str().graphemes(true)); // Iterate over values of array.
iter!(for pattern in array);
} }
(_, Value::Dict(dict)) => { (_, Value::Dict(dict)) => {
// Iterate over pairs of dict. // Iterate over pairs of dict.
iter!(for pattern in dict.pairs()); iter!(for pattern in dict.pairs());
} }
(_, Value::Array(array)) => { (Pattern::Normal(_) | Pattern::Placeholder(_), Value::Str(str)) => {
// Iterate over values of array. // Iterate over graphemes of string.
iter!(for pattern in array); iter!(for pattern in str.as_str().graphemes(true));
} }
(ast::Pattern::Normal(_), _) => { (Pattern::Destructuring(_), Value::Str(_)) => {
bail!(pattern.span(), "cannot destructure values of {}", iterable_type);
}
_ => {
bail!(self.iterable().span(), "cannot loop over {}", iterable_type); bail!(self.iterable().span(), "cannot loop over {}", iterable_type);
} }
(_, _) => {
bail!(pattern.span(), "cannot destructure values of {}", iterable_type)
}
} }
if flow.is_some() { if flow.is_some() {

View File

@ -0,0 +1,49 @@
// Issue #3275: clearer errors for loops, https://github.com/typst/typst/issues/3275
// Ref: false
---
// Normal variable.
#for x in (1, 2) {}
#for x in (a: 1, b: 2) {}
#for x in "foo" {}
---
// Placeholder.
#for _ in (1, 2) {}
#for _ in (a: 1, b: 2) {}
#for _ in "foo" {}
---
// Destructuring.
#for (k, v) in (("a", 1), ("b", 2), ("c", 3)) {}
#for (k, ..) in (("a", 1), ("b", 2), ("c", 3)) {}
#for (k, v) in (a: 1, b: 2, c: 3) {}
#for (.., v) in (a: 1, b: 2, c: 3) {}
---
// Error: 11-17 cannot loop over content
#for x in [1, 2] {}
---
// Error: 11-25 cannot loop over bytes
#for _ in bytes((22, 0)) {}
---
// Error: 16-21 cannot loop over integer
#for (x, y) in 12306 {}
---
// Error: 16-22 cannot loop over content
#for (x, y) in [1, 2] {}
---
// Error: 6-12 cannot destructure values of string
#for (x, y) in "foo" {}
---
// Error: 6-12 cannot destructure string
#for (x, y) in ("foo", "bar") {}
---
// Error: 6-12 cannot destructure integer
#for (x, y) in (1, 2) {}