From 7c84634e3fae69c3daa4d65e4381a80516bc13d9 Mon Sep 17 00:00:00 2001 From: Solomon Date: Sat, 23 Nov 2024 14:42:00 -0700 Subject: [PATCH] return accurate type errors from blocks/expressions in type unions (#14420) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit # User-Facing Changes - `expected ` errors are now propagated from `Closure | Block | Expression` instead of falling back to "expected one of..." for the block: Before: ```nushell def foo [bar: bool] {} if true {} else { foo 1 } ────┬──── ╰── expected one of a list of accepted shapes: [Block, Expression] ``` After: ```nushell if true {} else { foo 1 } ┬ ╰── expected bool ``` --- crates/nu-parser/src/parser.rs | 119 ++++++++++++++------------------- tests/repl/test_type_check.rs | 18 +++++ 2 files changed, 67 insertions(+), 70 deletions(-) diff --git a/crates/nu-parser/src/parser.rs b/crates/nu-parser/src/parser.rs index 7db2e112bf..4048bec2a9 100644 --- a/crates/nu-parser/src/parser.rs +++ b/crates/nu-parser/src/parser.rs @@ -772,6 +772,51 @@ fn calculate_end_span( } } +fn parse_oneof( + working_set: &mut StateWorkingSet, + spans: &[Span], + spans_idx: &mut usize, + possible_shapes: &Vec, + multispan: bool, +) -> Expression { + for shape in possible_shapes { + let starting_error_count = working_set.parse_errors.len(); + let value = match multispan { + true => parse_multispan_value(working_set, spans, spans_idx, shape), + false => parse_value(working_set, spans[*spans_idx], shape), + }; + + if starting_error_count == working_set.parse_errors.len() { + return value; + } + + // while trying the possible shapes, ignore Expected type errors + // unless they're inside a block, closure, or expression + let propagate_error = match working_set.parse_errors.last() { + Some(ParseError::Expected(_, error_span)) + | Some(ParseError::ExpectedWithStringMsg(_, error_span)) => { + matches!( + shape, + SyntaxShape::Block | SyntaxShape::Closure(_) | SyntaxShape::Expression + ) && *error_span != spans[*spans_idx] + } + _ => true, + }; + if !propagate_error { + working_set.parse_errors.truncate(starting_error_count); + } + } + + if working_set.parse_errors.is_empty() { + working_set.error(ParseError::ExpectedWithStringMsg( + format!("one of a list of accepted shapes: {possible_shapes:?}"), + spans[*spans_idx], + )); + } + + Expression::garbage(working_set, spans[*spans_idx]) +} + pub fn parse_multispan_value( working_set: &mut StateWorkingSet, spans: &[Span], @@ -800,54 +845,10 @@ pub fn parse_multispan_value( arg } - SyntaxShape::OneOf(shapes) => { - // handle for `if` command. - //let block_then_exp = shapes.as_slice() == [SyntaxShape::Block, SyntaxShape::Expression]; - for shape in shapes.iter() { - let starting_error_count = working_set.parse_errors.len(); - let s = parse_multispan_value(working_set, spans, spans_idx, shape); - - if starting_error_count == working_set.parse_errors.len() { - return s; - } else if let Some( - ParseError::Expected(..) | ParseError::ExpectedWithStringMsg(..), - ) = working_set.parse_errors.last() - { - working_set.parse_errors.truncate(starting_error_count); - continue; - } - // `if` is parsing block first and then expression. - // when we're writing something like `else if $a`, parsing as a - // block will result to error(because it's not a block) - // - // If parse as a expression also failed, user is more likely concerned - // about expression failure rather than "expect block failure"". - - // FIXME FIXME FIXME - // if block_then_exp { - // match &err { - // Some(ParseError::Expected(expected, _)) => { - // if expected.starts_with("block") { - // err = e - // } - // } - // _ => err = err.or(e), - // } - // } else { - // err = err.or(e) - // } - } - let span = spans[*spans_idx]; - - if working_set.parse_errors.is_empty() { - working_set.error(ParseError::ExpectedWithStringMsg( - format!("one of a list of accepted shapes: {shapes:?}"), - span, - )); - } - - Expression::garbage(working_set, span) + SyntaxShape::OneOf(possible_shapes) => { + parse_oneof(working_set, spans, spans_idx, possible_shapes, true) } + SyntaxShape::Expression => { trace!("parsing: expression"); @@ -4826,29 +4827,7 @@ pub fn parse_value( SyntaxShape::ExternalArgument => parse_regular_external_arg(working_set, span), SyntaxShape::OneOf(possible_shapes) => { - for s in possible_shapes { - let starting_error_count = working_set.parse_errors.len(); - let value = parse_value(working_set, span, s); - - if starting_error_count == working_set.parse_errors.len() { - return value; - } else if let Some( - ParseError::Expected(..) | ParseError::ExpectedWithStringMsg(..), - ) = working_set.parse_errors.last() - { - working_set.parse_errors.truncate(starting_error_count); - continue; - } - } - - if working_set.parse_errors.is_empty() { - working_set.error(ParseError::ExpectedWithStringMsg( - format!("one of a list of accepted shapes: {possible_shapes:?}"), - span, - )); - } - - Expression::garbage(working_set, span) + parse_oneof(working_set, &[span], &mut 0, possible_shapes, false) } SyntaxShape::Any => { diff --git a/tests/repl/test_type_check.rs b/tests/repl/test_type_check.rs index 95ef7f4323..07a8d99495 100644 --- a/tests/repl/test_type_check.rs +++ b/tests/repl/test_type_check.rs @@ -1,4 +1,5 @@ use crate::repl::tests::{fail_test, run_test, TestResult}; +use rstest::rstest; #[test] fn chained_operator_typecheck() -> TestResult { @@ -154,3 +155,20 @@ fn in_variable_expression_wrong_output_type() -> TestResult { "expected int", ) } + +#[rstest] +#[case("if true {} else { foo 1 }")] +#[case("if true {} else if (foo 1) == null { }")] +#[case("match 1 { 0 => { foo 1 } }")] +#[case("try { } catch { foo 1 }")] +/// type errors should propagate from `OneOf(Block | Closure | Expression, ..)` +fn in_oneof_block_expected_type(#[case] input: &str) -> TestResult { + let def = "def foo [bar: bool] {};"; + + fail_test(&format!("{def} {input}"), "expected bool") +} + +#[test] +fn in_oneof_block_expected_block() -> TestResult { + fail_test("match 1 { 0 => { try 3 } }", "expected block") +}