// 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 insta::assert_snapshot; use itertools::Itertools; use regex::Regex; use crate::common::TestEnvironment; #[test] fn test_config_list_single() { let test_env = TestEnvironment::default(); test_env.add_config( r#" [test-table] somekey = "some value" "#, ); let stdout = test_env.jj_cmd_success( test_env.env_root(), &["config", "list", "test-table.somekey"], ); insta::assert_snapshot!(stdout, @r###" test-table.somekey="some value" "###); let stdout = test_env.jj_cmd_success( test_env.env_root(), &["config", "list", r#"-Tname ++ "\n""#, "test-table.somekey"], ); insta::assert_snapshot!(stdout, @r###" test-table.somekey "###); } #[test] fn test_config_list_nonexistent() { let test_env = TestEnvironment::default(); let (stdout, stderr) = test_env.jj_cmd_ok( test_env.env_root(), &["config", "list", "nonexistent-test-key"], ); insta::assert_snapshot!(stdout, @""); insta::assert_snapshot!(stderr, @r###" Warning: No matching config key for nonexistent-test-key "###); } #[test] fn test_config_list_table() { let test_env = TestEnvironment::default(); test_env.add_config( r#" [test-table] x = true y.foo = "abc" y.bar = 123 "#, ); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "list", "test-table"]); insta::assert_snapshot!( stdout, @r###" test-table.x=true test-table.y.bar=123 test-table.y.foo="abc" "###); } #[test] fn test_config_list_array() { let test_env = TestEnvironment::default(); test_env.add_config( r#" test-array = [1, "b", 3.4] "#, ); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "list", "test-array"]); insta::assert_snapshot!(stdout, @r###" test-array=[1, "b", 3.4] "###); } #[test] fn test_config_list_inline_table() { let test_env = TestEnvironment::default(); test_env.add_config( r#" [[test-table]] x = 1 [[test-table]] y = ["z"] "#, ); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "list", "test-table"]); insta::assert_snapshot!(stdout, @r###" test-table=[{x=1}, {y=["z"]}] "###); } #[test] fn test_config_list_all() { let test_env = TestEnvironment::default(); test_env.add_config( r#" test-val = [1, 2, 3] [test-table] x = true y.foo = "abc" y.bar = 123 "#, ); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "list"]); insta::assert_snapshot!( find_stdout_lines(r"(test-val|test-table\b[^=]*)", &stdout), @r###" test-table.x=true test-table.y.bar=123 test-table.y.foo="abc" test-val=[1, 2, 3] "###); } #[test] fn test_config_list_layer() { let mut test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let user_config_path = test_env.config_path().join("config.toml"); test_env.set_config_path(user_config_path.to_owned()); let repo_path = test_env.env_root().join("repo"); // User test_env.jj_cmd_ok( &repo_path, &["config", "set", "--user", "test-key", "test-val"], ); test_env.jj_cmd_ok( &repo_path, &[ "config", "set", "--user", "test-layered-key", "test-original-val", ], ); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", "--user"]); insta::assert_snapshot!(stdout, @r###" test-key="test-val" test-layered-key="test-original-val" "###); // Repo test_env.jj_cmd_ok( &repo_path, &[ "config", "set", "--repo", "test-layered-key", "test-layered-val", ], ); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", "--user"]); insta::assert_snapshot!(stdout, @r###" test-key="test-val" "###); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", "--repo"]); insta::assert_snapshot!(stdout, @r###" test-layered-key="test-layered-val" "###); } #[test] fn test_config_layer_override_default() { 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 config_key = "merge-tools.vimdiff.program"; // Default let stdout = test_env.jj_cmd_success( &repo_path, &["config", "list", config_key, "--include-defaults"], ); insta::assert_snapshot!(stdout, @r###" merge-tools.vimdiff.program="vim" "###); // User test_env.add_config(&format!("{config_key} = {value:?}\n", value = "user")); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" merge-tools.vimdiff.program="user" "###); // Repo std::fs::write( repo_path.join(".jj/repo/config.toml"), format!("{config_key} = {value:?}\n", value = "repo"), ) .unwrap(); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" merge-tools.vimdiff.program="repo" "###); // Command argument let stdout = test_env.jj_cmd_success( &repo_path, &[ "config", "list", config_key, "--config-toml", &format!("{config_key}={value:?}", value = "command-arg"), ], ); insta::assert_snapshot!(stdout, @r###" merge-tools.vimdiff.program="command-arg" "###); // Allow printing overridden values let stdout = test_env.jj_cmd_success( &repo_path, &[ "config", "list", config_key, "--include-overridden", "--config-toml", &format!("{config_key}={value:?}", value = "command-arg"), ], ); insta::assert_snapshot!(stdout, @r###" # merge-tools.vimdiff.program="user" # merge-tools.vimdiff.program="repo" merge-tools.vimdiff.program="command-arg" "###); let stdout = test_env.jj_cmd_success( &repo_path, &[ "config", "list", "--color=always", config_key, "--include-overridden", ], ); insta::assert_snapshot!(stdout, @r###" # merge-tools.vimdiff.program="user" merge-tools.vimdiff.program="repo" "###); } #[test] fn test_config_layer_override_env() { let mut 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 config_key = "ui.editor"; // Environment base test_env.add_env_var("EDITOR", "env-base"); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" ui.editor="env-base" "###); // User test_env.add_config(&format!("{config_key} = {value:?}\n", value = "user")); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" ui.editor="user" "###); // Repo std::fs::write( repo_path.join(".jj/repo/config.toml"), format!("{config_key} = {value:?}\n", value = "repo"), ) .unwrap(); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" ui.editor="repo" "###); // Environment override test_env.add_env_var("JJ_EDITOR", "env-override"); let stdout = test_env.jj_cmd_success(&repo_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" ui.editor="env-override" "###); // Command argument let stdout = test_env.jj_cmd_success( &repo_path, &[ "config", "list", config_key, "--config-toml", &format!("{config_key}={value:?}", value = "command-arg"), ], ); insta::assert_snapshot!(stdout, @r###" ui.editor="command-arg" "###); // Allow printing overridden values let stdout = test_env.jj_cmd_success( &repo_path, &[ "config", "list", config_key, "--include-overridden", "--config-toml", &format!("{config_key}={value:?}", value = "command-arg"), ], ); insta::assert_snapshot!(stdout, @r###" # ui.editor="env-base" # ui.editor="user" # ui.editor="repo" # ui.editor="env-override" ui.editor="command-arg" "###); } #[test] fn test_config_layer_workspace() { let test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "main"]); let main_path = test_env.env_root().join("main"); let secondary_path = test_env.env_root().join("secondary"); let config_key = "ui.editor"; std::fs::write(main_path.join("file"), "contents").unwrap(); test_env.jj_cmd_ok(&main_path, &["new"]); test_env.jj_cmd_ok( &main_path, &["workspace", "add", "--name", "second", "../secondary"], ); // Repo std::fs::write( main_path.join(".jj/repo/config.toml"), format!("{config_key} = {value:?}\n", value = "main-repo"), ) .unwrap(); let stdout = test_env.jj_cmd_success(&main_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" ui.editor="main-repo" "###); let stdout = test_env.jj_cmd_success(&secondary_path, &["config", "list", config_key]); insta::assert_snapshot!(stdout, @r###" ui.editor="main-repo" "###); } #[test] fn test_config_set_missing_opts() { let test_env = TestEnvironment::default(); let stderr = test_env.jj_cmd_cli_error(test_env.env_root(), &["config", "set"]); insta::assert_snapshot!(stderr, @r###" error: the following required arguments were not provided: <--user|--repo> Usage: jj config set <--user|--repo> For more information, try '--help'. "###); } #[test] fn test_config_set_for_user() { let mut test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); // Point to a config file since `config set` can't handle directories. let user_config_path = test_env.config_path().join("config.toml"); test_env.set_config_path(user_config_path.to_owned()); let repo_path = test_env.env_root().join("repo"); test_env.jj_cmd_ok( &repo_path, &["config", "set", "--user", "test-key", "test-val"], ); test_env.jj_cmd_ok( &repo_path, &["config", "set", "--user", "test-table.foo", "true"], ); // Ensure test-key successfully written to user config. let user_config_toml = std::fs::read_to_string(&user_config_path) .unwrap_or_else(|_| panic!("Failed to read file {}", user_config_path.display())); insta::assert_snapshot!(user_config_toml, @r###" test-key = "test-val" [test-table] foo = true "###); } #[test] fn test_config_set_for_repo() { 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.jj_cmd_ok( &repo_path, &["config", "set", "--repo", "test-key", "test-val"], ); test_env.jj_cmd_ok( &repo_path, &["config", "set", "--repo", "test-table.foo", "true"], ); // Ensure test-key successfully written to user config. let expected_repo_config_path = repo_path.join(".jj/repo/config.toml"); let repo_config_toml = std::fs::read_to_string(&expected_repo_config_path).unwrap_or_else(|_| { panic!( "Failed to read file {}", expected_repo_config_path.display() ) }); insta::assert_snapshot!(repo_config_toml, @r###" test-key = "test-val" [test-table] foo = true "###); } #[test] fn test_config_set_toml_types() { let mut test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let user_config_path = test_env.config_path().join("config.toml"); test_env.set_config_path(user_config_path.clone()); let repo_path = test_env.env_root().join("repo"); let set_value = |key, value| { test_env.jj_cmd_success(&repo_path, &["config", "set", "--user", key, value]); }; set_value("test-table.integer", "42"); set_value("test-table.float", "3.14"); set_value("test-table.array", r#"["one", "two"]"#); set_value("test-table.boolean", "true"); set_value("test-table.string", r#""foo""#); set_value("test-table.invalid", r"a + b"); insta::assert_snapshot!(std::fs::read_to_string(&user_config_path).unwrap(), @r###" [test-table] integer = 42 float = 3.14 array = ["one", "two"] boolean = true string = "foo" invalid = "a + b" "###); } #[test] fn test_config_set_type_mismatch() { let mut test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let user_config_path = test_env.config_path().join("config.toml"); test_env.set_config_path(user_config_path); let repo_path = test_env.env_root().join("repo"); test_env.jj_cmd_ok( &repo_path, &["config", "set", "--user", "test-table.foo", "test-val"], ); let stderr = test_env.jj_cmd_failure( &repo_path, &["config", "set", "--user", "test-table", "not-a-table"], ); insta::assert_snapshot!(stderr, @r###" Error: Failed to set test-table: would overwrite entire table "###); // But it's fine to overwrite arrays and inline tables test_env.jj_cmd_success( &repo_path, &["config", "set", "--user", "test-table.array", "[1,2,3]"], ); test_env.jj_cmd_success( &repo_path, &["config", "set", "--user", "test-table.array", "[4,5,6]"], ); test_env.jj_cmd_success( &repo_path, &["config", "set", "--user", "test-table.inline", "{ x = 42}"], ); test_env.jj_cmd_success( &repo_path, &["config", "set", "--user", "test-table.inline", "42"], ); } #[test] fn test_config_set_nontable_parent() { let mut test_env = TestEnvironment::default(); test_env.jj_cmd_ok(test_env.env_root(), &["git", "init", "repo"]); let user_config_path = test_env.config_path().join("config.toml"); test_env.set_config_path(user_config_path); let repo_path = test_env.env_root().join("repo"); test_env.jj_cmd_ok( &repo_path, &["config", "set", "--user", "test-nontable", "test-val"], ); let stderr = test_env.jj_cmd_failure( &repo_path, &["config", "set", "--user", "test-nontable.foo", "test-val"], ); insta::assert_snapshot!(stderr, @r###" Error: Failed to set test-nontable.foo: would overwrite non-table value with parent table "###); } #[test] fn test_config_edit_missing_opt() { let test_env = TestEnvironment::default(); let stderr = test_env.jj_cmd_cli_error(test_env.env_root(), &["config", "edit"]); insta::assert_snapshot!(stderr, @r###" error: the following required arguments were not provided: <--user|--repo> Usage: jj config edit <--user|--repo> For more information, try '--help'. "###); } #[test] fn test_config_edit_user() { let mut 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 edit_script = test_env.set_up_fake_editor(); std::fs::write( edit_script, format!("expectpath\n{}", test_env.config_path().to_str().unwrap()), ) .unwrap(); test_env.jj_cmd_ok(&repo_path, &["config", "edit", "--user"]); } #[test] fn test_config_edit_repo() { let mut 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 edit_script = test_env.set_up_fake_editor(); std::fs::write( edit_script, format!( "expectpath\n{}", repo_path.join(".jj/repo/config.toml").to_str().unwrap() ), ) .unwrap(); test_env.jj_cmd_ok(&repo_path, &["config", "edit", "--repo"]); } #[test] fn test_config_path() { 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"); assert_snapshot!( test_env.jj_cmd_success(&repo_path, &["config", "path", "--user"]), @r###" $TEST_ENV/config "### ); assert_snapshot!( test_env.jj_cmd_success(&repo_path, &["config", "path", "--repo"]), @r###" $TEST_ENV/repo/.jj/repo/config.toml "### ); } #[test] fn test_config_edit_repo_outside_repo() { let test_env = TestEnvironment::default(); let stderr = test_env.jj_cmd_failure(test_env.env_root(), &["config", "edit", "--repo"]); insta::assert_snapshot!(stderr, @r###" Error: There is no jj repo in "." "###); } #[test] fn test_config_get() { let test_env = TestEnvironment::default(); test_env.add_config( r#" [table] string = "some value 1" int = 123 list = ["list", "value"] overridden = "foo" "#, ); test_env.add_config( r#" [table] overridden = "bar" "#, ); let stdout = test_env.jj_cmd_failure(test_env.env_root(), &["config", "get", "nonexistent"]); insta::assert_snapshot!(stdout, @r###" Config error: configuration property "nonexistent" not found For help, see https://github.com/martinvonz/jj/blob/main/docs/config.md. "###); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "get", "table.string"]); insta::assert_snapshot!(stdout, @r###" some value 1 "###); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "get", "table.int"]); insta::assert_snapshot!(stdout, @r###" 123 "###); let stdout = test_env.jj_cmd_failure(test_env.env_root(), &["config", "get", "table.list"]); insta::assert_snapshot!(stdout, @r###" Config error: invalid type: sequence, expected a value convertible to a string For help, see https://github.com/martinvonz/jj/blob/main/docs/config.md. "###); let stdout = test_env.jj_cmd_failure(test_env.env_root(), &["config", "get", "table"]); insta::assert_snapshot!(stdout, @r###" Config error: invalid type: map, expected a value convertible to a string For help, see https://github.com/martinvonz/jj/blob/main/docs/config.md. "###); let stdout = test_env.jj_cmd_success(test_env.env_root(), &["config", "get", "table.overridden"]); insta::assert_snapshot!(stdout, @"bar"); } fn find_stdout_lines(keyname_pattern: &str, stdout: &str) -> String { let key_line_re = Regex::new(&format!(r"(?m)^{keyname_pattern}=.*$")).unwrap(); key_line_re .find_iter(stdout) .map(|m| m.as_str()) .collect_vec() .join("\n") }