// Copyright 2022 The Jujutsu Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use std::path::Path; use indoc::indoc; use crate::common::TestEnvironment; #[test] fn test_templater_parse_error() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let repo_path = test_env.env_root().join("repo"); let render_err = |template| test_env.jj_cmd_failure(&repo_path, &["log", "-T", template]); insta::assert_snapshot!(render_err(r#"description ()"#), @r#" Error: Failed to parse template: Syntax error Caused by: --> 1:13 | 1 | description () | ^--- | = expected , `++`, `||`, `&&`, `==`, `!=`, `>=`, `>`, `<=`, or `<` "#); // Typo test_env.add_config( r###" [template-aliases] 'conflicting' = '' 'shorted()' = '' 'socat(x)' = 'x' 'format_id(id)' = 'id.sort()' "###, ); insta::assert_snapshot!(render_err(r#"conflicts"#), @r###" Error: Failed to parse template: Keyword "conflicts" doesn't exist Caused by: --> 1:1 | 1 | conflicts | ^-------^ | = Keyword "conflicts" doesn't exist Hint: Did you mean "conflict", "conflicting"? "###); insta::assert_snapshot!(render_err(r#"commit_id.shorter()"#), @r###" Error: Failed to parse template: Method "shorter" doesn't exist for type "CommitOrChangeId" Caused by: --> 1:11 | 1 | commit_id.shorter() | ^-----^ | = Method "shorter" doesn't exist for type "CommitOrChangeId" Hint: Did you mean "short", "shortest"? "###); insta::assert_snapshot!(render_err(r#"oncat()"#), @r###" Error: Failed to parse template: Function "oncat" doesn't exist Caused by: --> 1:1 | 1 | oncat() | ^---^ | = Function "oncat" doesn't exist Hint: Did you mean "concat", "socat"? "###); insta::assert_snapshot!(render_err(r#""".lines().map(|s| se)"#), @r###" Error: Failed to parse template: Keyword "se" doesn't exist Caused by: --> 1:20 | 1 | "".lines().map(|s| se) | ^^ | = Keyword "se" doesn't exist Hint: Did you mean "s", "self"? "###); insta::assert_snapshot!(render_err(r#"format_id(commit_id)"#), @r#" Error: Failed to parse template: In alias "format_id(id)" Caused by: 1: --> 1:1 | 1 | format_id(commit_id) | ^------------------^ | = In alias "format_id(id)" 2: --> 1:4 | 1 | id.sort() | ^--^ | = Method "sort" doesn't exist for type "CommitOrChangeId" Hint: Did you mean "short", "shortest"? "#); // -Tbuiltin shows the predefined builtin_* aliases. This isn't 100% // guaranteed, but is nice. insta::assert_snapshot!(render_err(r#"builtin"#), @r#" Error: Failed to parse template: Keyword "builtin" doesn't exist Caused by: --> 1:1 | 1 | builtin | ^-----^ | = Keyword "builtin" doesn't exist Hint: Did you mean "builtin_log_comfortable", "builtin_log_compact", "builtin_log_compact_full_description", "builtin_log_detailed", "builtin_log_node", "builtin_log_node_ascii", "builtin_log_oneline", "builtin_op_log_comfortable", "builtin_op_log_compact", "builtin_op_log_node", "builtin_op_log_node_ascii"? "#); } #[test] fn test_template_parse_warning() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let repo_path = test_env.env_root().join("repo"); let template = indoc! {r#" separate(' ', branches, local_branches, remote_branches, self.contained_in('branches()'), ) "#}; let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["log", "-r@", "-T", template]); insta::assert_snapshot!(stdout, @r#" @ false │ ~ "#); insta::assert_snapshot!(stderr, @r#" Warning: In template expression --> 2:3 | 2 | branches, | ^------^ | = branches() is deprecated; use bookmarks() instead Warning: In template expression --> 3:3 | 3 | local_branches, | ^------------^ | = local_branches() is deprecated; use local_bookmarks() instead Warning: In template expression --> 4:3 | 4 | remote_branches, | ^-------------^ | = remote_branches() is deprecated; use remote_bookmarks() instead Warning: In template expression --> 5:21 | 5 | self.contained_in('branches()'), | ^----------^ | = In revset expression --> 1:1 | 1 | branches() | ^------^ | = branches() is deprecated; use bookmarks() instead "#); } #[test] fn test_templater_upper_lower() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let repo_path = test_env.env_root().join("repo"); let render = |template| get_colored_template_output(&test_env, &repo_path, "@-", template); insta::assert_snapshot!( render(r#"change_id.shortest(4).upper() ++ change_id.shortest(4).upper().lower()"#), @"ZZZZzzzz"); insta::assert_snapshot!( render(r#""Hello".upper() ++ "Hello".lower()"#), @"HELLOhello"); } #[test] fn test_templater_alias() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let repo_path = test_env.env_root().join("repo"); let render = |template| get_template_output(&test_env, &repo_path, "@-", template); let render_err = |template| test_env.jj_cmd_failure(&repo_path, &["log", "-T", template]); test_env.add_config( r###" [template-aliases] 'my_commit_id' = 'commit_id.short()' 'syntax_error' = 'foo.' 'name_error' = 'unknown_id' 'recurse' = 'recurse1' 'recurse1' = 'recurse2()' 'recurse2()' = 'recurse' 'identity(x)' = 'x' 'coalesce(x, y)' = 'if(x, x, y)' 'deprecated()' = 'branches ++ self.contained_in("branches()")' 'builtin_log_node' = '"#"' 'builtin_op_log_node' = '"#"' "###, ); insta::assert_snapshot!(render("my_commit_id"), @"000000000000"); insta::assert_snapshot!(render("identity(my_commit_id)"), @"000000000000"); insta::assert_snapshot!(render_err("commit_id ++ syntax_error"), @r#" Error: Failed to parse template: In alias "syntax_error" Caused by: 1: --> 1:14 | 1 | commit_id ++ syntax_error | ^----------^ | = In alias "syntax_error" 2: --> 1:5 | 1 | foo. | ^--- | = expected "#); insta::assert_snapshot!(render_err("commit_id ++ name_error"), @r#" Error: Failed to parse template: In alias "name_error" Caused by: 1: --> 1:14 | 1 | commit_id ++ name_error | ^--------^ | = In alias "name_error" 2: --> 1:1 | 1 | unknown_id | ^--------^ | = Keyword "unknown_id" doesn't exist "#); insta::assert_snapshot!(render_err(r#"identity(identity(commit_id.short("")))"#), @r#" Error: Failed to parse template: In alias "identity(x)" Caused by: 1: --> 1:1 | 1 | identity(identity(commit_id.short(""))) | ^-------------------------------------^ | = In alias "identity(x)" 2: --> 1:1 | 1 | x | ^ | = In function parameter "x" 3: --> 1:10 | 1 | identity(identity(commit_id.short(""))) | ^---------------------------^ | = In alias "identity(x)" 4: --> 1:1 | 1 | x | ^ | = In function parameter "x" 5: --> 1:35 | 1 | identity(identity(commit_id.short(""))) | ^^ | = Expected expression of type "Integer", but actual type is "String" "#); insta::assert_snapshot!(render_err("commit_id ++ recurse"), @r#" Error: Failed to parse template: In alias "recurse" Caused by: 1: --> 1:14 | 1 | commit_id ++ recurse | ^-----^ | = In alias "recurse" 2: --> 1:1 | 1 | recurse1 | ^------^ | = In alias "recurse1" 3: --> 1:1 | 1 | recurse2() | ^--------^ | = In alias "recurse2()" 4: --> 1:1 | 1 | recurse | ^-----^ | = Alias "recurse" expanded recursively "#); insta::assert_snapshot!(render_err("identity()"), @r###" Error: Failed to parse template: Function "identity": Expected 1 arguments Caused by: --> 1:10 | 1 | identity() | ^ | = Function "identity": Expected 1 arguments "###); insta::assert_snapshot!(render_err("identity(commit_id, commit_id)"), @r###" Error: Failed to parse template: Function "identity": Expected 1 arguments Caused by: --> 1:10 | 1 | identity(commit_id, commit_id) | ^------------------^ | = Function "identity": Expected 1 arguments "###); insta::assert_snapshot!(render_err(r#"coalesce(label("x", "not boolean"), "")"#), @r#" Error: Failed to parse template: In alias "coalesce(x, y)" Caused by: 1: --> 1:1 | 1 | coalesce(label("x", "not boolean"), "") | ^-------------------------------------^ | = In alias "coalesce(x, y)" 2: --> 1:4 | 1 | if(x, x, y) | ^ | = In function parameter "x" 3: --> 1:10 | 1 | coalesce(label("x", "not boolean"), "") | ^-----------------------^ | = Expected expression of type "Boolean", but actual type is "Template" "#); insta::assert_snapshot!(render_err("(-my_commit_id)"), @r#" Error: Failed to parse template: In alias "my_commit_id" Caused by: 1: --> 1:3 | 1 | (-my_commit_id) | ^----------^ | = In alias "my_commit_id" 2: --> 1:1 | 1 | commit_id.short() | ^---------------^ | = Expected expression of type "Integer", but actual type is "String" "#); let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["log", "-r@", "-Tdeprecated()"]); insta::assert_snapshot!(stdout, @r##" # false │ ~ "##); insta::assert_snapshot!(stderr, @r#" Warning: In template expression --> 1:1 | 1 | deprecated() | ^----------^ | = In alias "deprecated()" --> 1:1 | 1 | branches ++ self.contained_in("branches()") | ^------^ | = branches() is deprecated; use bookmarks() instead Warning: In template expression --> 1:1 | 1 | deprecated() | ^----------^ | = In alias "deprecated()" --> 1:31 | 1 | branches ++ self.contained_in("branches()") | ^----------^ | = In revset expression --> 1:1 | 1 | branches() | ^------^ | = branches() is deprecated; use bookmarks() instead "#); } #[test] fn test_templater_alias_override() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let repo_path = test_env.env_root().join("repo"); test_env.add_config( r#" [template-aliases] 'f(x)' = '"user"' "#, ); // 'f(x)' should be overridden by --config-toml 'f(a)'. If aliases were sorted // purely by name, 'f(a)' would come first. let stdout = test_env.jj_cmd_success( &repo_path, &[ "log", "--no-graph", "-r@", "-T", r#"f(_)"#, "--config-toml", r#"template-aliases.'f(a)' = '"arg"'"#, ], ); insta::assert_snapshot!(stdout, @"arg"); } #[test] fn test_templater_bad_alias_decl() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let repo_path = test_env.env_root().join("repo"); test_env.add_config( r###" [template-aliases] 'badfn(a, a)' = 'a' 'my_commit_id' = 'commit_id.short()' "###, ); // Invalid declaration should be warned and ignored. let (stdout, stderr) = test_env.jj_cmd_ok(&repo_path, &["log", "--no-graph", "-r@-", "-Tmy_commit_id"]); insta::assert_snapshot!(stdout, @"000000000000"); insta::assert_snapshot!(stderr, @r###" Warning: Failed to load "template-aliases.badfn(a, a)": --> 1:7 | 1 | badfn(a, a) | ^--^ | = Redefinition of function parameter "###); } fn get_template_output( test_env: &TestEnvironment, repo_path: &Path, rev: &str, template: &str, ) -> String { test_env.jj_cmd_success(repo_path, &["log", "--no-graph", "-r", rev, "-T", template]) } fn get_colored_template_output( test_env: &TestEnvironment, repo_path: &Path, rev: &str, template: &str, ) -> String { test_env.jj_cmd_success( repo_path, &[ "log", "--color=always", "--no-graph", "-r", rev, "-T", template, ], ) }