From b44148871a0db33fb442259bdf7ce1048d35fa7b Mon Sep 17 00:00:00 2001 From: Yuya Nishihara Date: Sun, 12 Feb 2023 21:44:33 +0900 Subject: [PATCH] templater: save alias chain to report type/name error in original context Since type/name checking is made after alias substitution, we need to preserve the original context to generate a readable error message. We could instead attach a stack of (alias_id, span) to ExpressionNode, but the extra AliasExpanded node helps to capture downstream error by a single .map_err() call. --- src/template_parser.rs | 18 ++++++++++++++++-- tests/test_templater.rs | 38 +++++++++++++++++++++++++++++++++++--- 2 files changed, 51 insertions(+), 5 deletions(-) diff --git a/src/template_parser.rs b/src/template_parser.rs index ff5e10ed9..de31122e2 100644 --- a/src/template_parser.rs +++ b/src/template_parser.rs @@ -225,6 +225,8 @@ enum ExpressionKind<'i> { List(Vec>), FunctionCall(FunctionCallNode<'i>), MethodCall(MethodCallNode<'i>), + /// Identity node to preserve the span in the source template text. + AliasExpanded(TemplateAliasId<'i>, Box>), } #[derive(Clone, Debug, PartialEq)] @@ -464,10 +466,10 @@ fn expand_aliases<'i>( } fn expand_defn<'i>( - id: TemplateAliasId<'_>, + id: TemplateAliasId<'i>, defn: &'i str, locals: &HashMap<&str, ExpressionNode<'i>>, - span: pest::Span<'_>, + span: pest::Span<'i>, state: State<'_, 'i>, ) -> TemplateParseResult> { // The stack should be short, so let's simply do linear search and duplicate. @@ -487,6 +489,9 @@ fn expand_aliases<'i>( // Parsed defn could be cached if needed. parse_template(defn) .and_then(|node| expand_node(node, expanding_state)) + .map(|node| { + ExpressionNode::new(ExpressionKind::AliasExpanded(id, Box::new(node)), span) + }) .map_err(|e| e.within_alias_expansion(id, span)) } @@ -559,6 +564,12 @@ fn expand_aliases<'i>( }); Ok(node) } + ExpressionKind::AliasExpanded(id, subst) => { + // Just in case the original tree contained AliasExpanded node. + let subst = Box::new(expand_node(*subst, state)?); + node.kind = ExpressionKind::AliasExpanded(id, subst); + Ok(node) + } } } @@ -1088,6 +1099,8 @@ fn build_expression<'a, C: 'a>( } ExpressionKind::FunctionCall(function) => build_global_function(function, build_keyword), ExpressionKind::MethodCall(method) => build_method_call(method, build_keyword), + ExpressionKind::AliasExpanded(id, subst) => build_expression(subst, build_keyword) + .map_err(|e| e.within_alias_expansion(*id, node.span)), } } @@ -1181,6 +1194,7 @@ mod tests { let function = normalize_function_call(method.function); ExpressionKind::MethodCall(MethodCallNode { object, function }) } + ExpressionKind::AliasExpanded(_, subst) => normalize_tree(*subst).kind, }; ExpressionNode { kind: normalized_kind, diff --git a/tests/test_templater.rs b/tests/test_templater.rs index b3dad46e7..93f7e739f 100644 --- a/tests/test_templater.rs +++ b/tests/test_templater.rs @@ -428,9 +428,14 @@ fn test_templater_alias() { = expected identifier "###); - // TODO: outer template substitution should be reported too insta::assert_snapshot!(render_err("commit_id name_error"), @r###" - Error: Failed to parse template: --> 1:1 + Error: Failed to parse template: --> 1:11 + | + 1 | commit_id name_error + | ^--------^ + | + = Alias "name_error" cannot be expanded + --> 1:1 | 1 | unknown_id | ^--------^ @@ -438,6 +443,27 @@ fn test_templater_alias() { = Keyword "unknown_id" doesn't exist "###); + insta::assert_snapshot!(render_err(r#"identity(identity(commit_id.short("")))"#), @r###" + Error: Failed to parse template: --> 1:1 + | + 1 | identity(identity(commit_id.short(""))) + | ^-------------------------------------^ + | + = Alias "identity()" cannot be expanded + --> 1:10 + | + 1 | identity(identity(commit_id.short(""))) + | ^---------------------------^ + | + = Alias "identity()" cannot be expanded + --> 1:35 + | + 1 | identity(identity(commit_id.short(""))) + | ^^ + | + = Expected argument of type "Integer" + "###); + insta::assert_snapshot!(render_err("commit_id recurse"), @r###" Error: Failed to parse template: --> 1:11 | @@ -483,7 +509,13 @@ fn test_templater_alias() { "###); insta::assert_snapshot!(render_err(r#"coalesce(label("x", "not boolean"), "")"#), @r###" - Error: Failed to parse template: --> 1:10 + Error: Failed to parse template: --> 1:1 + | + 1 | coalesce(label("x", "not boolean"), "") + | ^-------------------------------------^ + | + = Alias "coalesce()" cannot be expanded + --> 1:10 | 1 | coalesce(label("x", "not boolean"), "") | ^-----------------------^