Isolate tests from user config (#12437)

# Description
This is an attempt to isolate the unit tests from whatever might be in
the user's config. If the
user's config is broken in some way or incompatible with this version
(for example, especially if
there are plugins that aren't built for this version), tests can
spuriously fail.

This makes tests more reliably pass the same way they would on CI even
if the user has config, and
should also make them run faster.

I think this is _good enough_, but I still think we should have a
specific config dir env variable for nushell specifically (rather than
having to use `XDG_CONFIG_HOME`, which would mess with other things) and
then we can just have `nu-test-support` set that to a temporary dir
containing the shipped default config files.

# Tests + Formatting
- 🟢 `toolkit fmt`
- 🟢 `toolkit clippy`
- 🟢 `toolkit test`
- 🟢 `toolkit test stdlib`
This commit is contained in:
Devyn Cairns 2024-04-09 15:27:46 -07:00 committed by GitHub
parent d7ba8872bf
commit d735607ac8
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
13 changed files with 105 additions and 82 deletions

View File

@ -47,7 +47,7 @@ fn ignore_shell_errors_works_for_external_with_semicolon() {
#[test] #[test]
fn ignore_program_errors_works_for_external_with_semicolon() { fn ignore_program_errors_works_for_external_with_semicolon() {
let actual = nu!(r#"do -p { nu -c 'exit 1' }; "text""#); let actual = nu!(r#"do -p { nu -n -c 'exit 1' }; "text""#);
assert_eq!(actual.err, ""); assert_eq!(actual.err, "");
assert_eq!(actual.out, "text"); assert_eq!(actual.out, "text");

View File

@ -7,7 +7,7 @@ fn basic_exec() {
let actual = nu!( let actual = nu!(
cwd: dirs.test(), pipeline( cwd: dirs.test(), pipeline(
r#" r#"
nu -c 'exec nu --testbin cococo a b c' nu -n -c 'exec nu --testbin cococo a b c'
"# "#
)); ));
@ -21,7 +21,7 @@ fn exec_complex_args() {
let actual = nu!( let actual = nu!(
cwd: dirs.test(), pipeline( cwd: dirs.test(), pipeline(
r#" r#"
nu -c 'exec nu --testbin cococo b --bar=2 -sab --arwr - -DTEEE=aasd-290 -90 --' nu -n -c 'exec nu --testbin cococo b --bar=2 -sab --arwr - -DTEEE=aasd-290 -90 --'
"# "#
)); ));
@ -35,7 +35,7 @@ fn exec_fail_batched_short_args() {
let actual = nu!( let actual = nu!(
cwd: dirs.test(), pipeline( cwd: dirs.test(), pipeline(
r#" r#"
nu -c 'exec nu --testbin cococo -ab 10' nu -n -c 'exec nu --testbin cococo -ab 10'
"# "#
)); ));
@ -49,7 +49,7 @@ fn exec_misc_values() {
let actual = nu!( let actual = nu!(
cwd: dirs.test(), pipeline( cwd: dirs.test(), pipeline(
r#" r#"
nu -c 'let x = "abc"; exec nu --testbin cococo $x ...[ a b c ]' nu -n -c 'let x = "abc"; exec nu --testbin cococo $x ...[ a b c ]'
"# "#
)); ));

View File

@ -3,8 +3,8 @@ use nu_test_support::nu;
#[test] #[test]
fn interleave_external_commands() { fn interleave_external_commands() {
let result = nu!("interleave \ let result = nu!("interleave \
{ nu -c 'print hello; print world' | lines | each { 'greeter: ' ++ $in } } \ { nu -n -c 'print hello; print world' | lines | each { 'greeter: ' ++ $in } } \
{ nu -c 'print nushell; print rocks' | lines | each { 'evangelist: ' ++ $in } } | \ { nu -n -c 'print nushell; print rocks' | lines | each { 'evangelist: ' ++ $in } } | \
each { print }; null"); each { print }; null");
assert!(result.out.contains("greeter: hello"), "{}", result.out); assert!(result.out.contains("greeter: hello"), "{}", result.out);
assert!(result.out.contains("greeter: world"), "{}", result.out); assert!(result.out.contains("greeter: world"), "{}", result.out);

View File

@ -18,7 +18,7 @@ fn early_return_if_false() {
fn return_works_in_script_without_def_main() { fn return_works_in_script_without_def_main() {
let actual = nu!( let actual = nu!(
cwd: "tests/fixtures/formats", pipeline( cwd: "tests/fixtures/formats", pipeline(
"nu early_return.nu" "nu -n early_return.nu"
)); ));
assert!(actual.err.is_empty()); assert!(actual.err.is_empty());
@ -28,7 +28,7 @@ fn return_works_in_script_without_def_main() {
fn return_works_in_script_with_def_main() { fn return_works_in_script_with_def_main() {
let actual = nu!( let actual = nu!(
cwd: "tests/fixtures/formats", cwd: "tests/fixtures/formats",
pipeline("nu early_return_outside_main.nu") pipeline("nu -n early_return_outside_main.nu")
); );
assert!(actual.err.is_empty()); assert!(actual.err.is_empty());
} }

View File

@ -93,7 +93,7 @@ fn save_stderr_and_stdout_to_afame_file() {
r#" r#"
$env.FOO = "bar"; $env.FOO = "bar";
$env.BAZ = "ZZZ"; $env.BAZ = "ZZZ";
do -c {nu -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -r save_test_5/new-file.txt --stderr save_test_5/new-file.txt do -c {nu -n -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -r save_test_5/new-file.txt --stderr save_test_5/new-file.txt
"#, "#,
); );
assert!(actual assert!(actual
@ -115,7 +115,7 @@ fn save_stderr_and_stdout_to_diff_file() {
r#" r#"
$env.FOO = "bar"; $env.FOO = "bar";
$env.BAZ = "ZZZ"; $env.BAZ = "ZZZ";
do -c {nu -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -r save_test_6/log.txt --stderr save_test_6/err.txt do -c {nu -n -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -r save_test_6/log.txt --stderr save_test_6/err.txt
"#, "#,
); );
@ -208,7 +208,7 @@ fn save_append_works_on_stderr() {
r#" r#"
$env.FOO = " New"; $env.FOO = " New";
$env.BAZ = " New Err"; $env.BAZ = " New Err";
do -i {nu -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -a -r save_test_11/log.txt --stderr save_test_11/err.txt"#, do -i {nu -n -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -a -r save_test_11/log.txt --stderr save_test_11/err.txt"#,
); );
let actual = file_contents(expected_file); let actual = file_contents(expected_file);
@ -229,7 +229,7 @@ fn save_not_overrides_err_by_default() {
r#" r#"
$env.FOO = " New"; $env.FOO = " New";
$env.BAZ = " New Err"; $env.BAZ = " New Err";
do -i {nu -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -r save_test_12/log.txt --stderr save_test_12/err.txt"#, do -i {nu -n -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -r save_test_12/log.txt --stderr save_test_12/err.txt"#,
); );
assert!(actual.err.contains("Destination file already exists")); assert!(actual.err.contains("Destination file already exists"));
@ -252,7 +252,7 @@ fn save_override_works_stderr() {
r#" r#"
$env.FOO = "New"; $env.FOO = "New";
$env.BAZ = "New Err"; $env.BAZ = "New Err";
do -i {nu -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -f -r save_test_13/log.txt --stderr save_test_13/err.txt"#, do -i {nu -n -c 'nu --testbin echo_env FOO; nu --testbin echo_env_stderr BAZ'} | save -f -r save_test_13/log.txt --stderr save_test_13/err.txt"#,
); );
let actual = file_contents(expected_file); let actual = file_contents(expected_file);

View File

@ -6,9 +6,9 @@ def run [
--short --short
] { ] {
if $short { if $short {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log ($message_level) --short "test message"' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log ($message_level) --short "test message"'
} else { } else {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log ($message_level) "test message"' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log ($message_level) "test message"'
} }
| complete | get --ignore-errors stderr | complete | get --ignore-errors stderr
} }

View File

@ -12,12 +12,12 @@ def run-command [
] { ] {
if ($level_prefix | is-empty) { if ($level_prefix | is-empty) {
if ($ansi | is-empty) { if ($ansi | is-empty) {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log custom "($message)" "($format)" ($log_level)' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log custom "($message)" "($format)" ($log_level)'
} else { } else {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log custom "($message)" "($format)" ($log_level) --ansi "($ansi)"' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log custom "($message)" "($format)" ($log_level) --ansi "($ansi)"'
} }
} else { } else {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log custom "($message)" "($format)" ($log_level) --level-prefix "($level_prefix)" --ansi "($ansi)"' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log custom "($message)" "($format)" ($log_level) --level-prefix "($level_prefix)" --ansi "($ansi)"'
} }
| complete | get --ignore-errors stderr | complete | get --ignore-errors stderr
} }

View File

@ -10,9 +10,9 @@ def run-command [
--short --short
] { ] {
if $short { if $short {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log ($message_level) --format "($format)" --short "($message)"' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log ($message_level) --format "($format)" --short "($message)"'
} else { } else {
^$nu.current-exe --commands $'use std; NU_log-level=($system_level) std log ($message_level) --format "($format)" "($message)"' ^$nu.current-exe --no-config-file --commands $'use std; NU_log-level=($system_level) std log ($message_level) --format "($format)" "($message)"'
} }
| complete | get --ignore-errors stderr | complete | get --ignore-errors stderr
} }

View File

@ -255,8 +255,8 @@ pub fn nu_run_test(opts: NuOpts, commands: impl AsRef<str>, with_std: bool) -> O
command command
.env(nu_utils::locale::LOCALE_OVERRIDE_ENV_VAR, locale) .env(nu_utils::locale::LOCALE_OVERRIDE_ENV_VAR, locale)
.env(NATIVE_PATH_ENV_VAR, paths_joined); .env(NATIVE_PATH_ENV_VAR, paths_joined);
// TODO: consider adding custom plugin path for tests to // Ensure that the user's config doesn't interfere with the tests
// not interfere with user local environment command.arg("--no-config-file");
if !with_std { if !with_std {
command.arg("--no-std-lib"); command.arg("--no-std-lib");
} }
@ -265,6 +265,9 @@ pub fn nu_run_test(opts: NuOpts, commands: impl AsRef<str>, with_std: bool) -> O
.stdout(Stdio::piped()) .stdout(Stdio::piped())
.stderr(Stdio::piped()); .stderr(Stdio::piped());
// Uncomment to debug the command being run:
// println!("=== command\n{command:?}\n");
let process = match command.spawn() { let process = match command.spawn() {
Ok(child) => child, Ok(child) => child,
Err(why) => panic!("Can't run test {:?} {}", crate::fs::executable_path(), why), Err(why) => panic!("Can't run test {:?} {}", crate::fs::executable_path(), why),
@ -293,8 +296,12 @@ pub fn nu_with_plugin_run_test(cwd: impl AsRef<Path>, plugins: &[&str], command:
}); });
let temp = tempdir().expect("couldn't create a temporary directory"); let temp = tempdir().expect("couldn't create a temporary directory");
let temp_plugin_file = temp.path().join("plugin.nu"); let [temp_config_file, temp_env_config_file, temp_plugin_file] =
std::fs::File::create(&temp_plugin_file).expect("couldn't create temporary plugin file"); ["config.nu", "env.nu", "plugin.nu"].map(|name| {
let temp_file = temp.path().join(name);
std::fs::File::create(&temp_file).expect("couldn't create temporary config file");
temp_file
});
crate::commands::ensure_plugins_built(); crate::commands::ensure_plugins_built();
@ -320,6 +327,10 @@ pub fn nu_with_plugin_run_test(cwd: impl AsRef<Path>, plugins: &[&str], command:
let process = match setup_command(&executable_path, &target_cwd) let process = match setup_command(&executable_path, &target_cwd)
.arg("--commands") .arg("--commands")
.arg(commands) .arg(commands)
.arg("--config")
.arg(temp_config_file)
.arg("--env-config")
.arg(temp_env_config_file)
.arg("--plugin-config") .arg("--plugin-config")
.arg(temp_plugin_file) .arg(temp_plugin_file)
.stdout(Stdio::piped()) .stdout(Stdio::piped())

View File

@ -146,8 +146,14 @@ pub(crate) fn run_file(
) -> Result<(), miette::ErrReport> { ) -> Result<(), miette::ErrReport> {
trace!("run_file"); trace!("run_file");
let mut stack = nu_protocol::engine::Stack::new(); let mut stack = nu_protocol::engine::Stack::new();
let start_time = std::time::Instant::now();
// if the --no-config-file(-n) option is NOT passed, load the plugin file,
// load the default env file or custom (depending on parsed_nu_cli_args.env_file),
// and maybe a custom config file (depending on parsed_nu_cli_args.config_file)
//
// if the --no-config-file(-n) flag is passed, do not load plugin, env, or config files
if parsed_nu_cli_args.no_config_file.is_none() {
let start_time = std::time::Instant::now();
#[cfg(feature = "plugin")] #[cfg(feature = "plugin")]
read_plugin_file( read_plugin_file(
engine_state, engine_state,
@ -167,7 +173,12 @@ pub(crate) fn run_file(
let start_time = std::time::Instant::now(); let start_time = std::time::Instant::now();
// only want to load config and env if relative argument is provided. // only want to load config and env if relative argument is provided.
if parsed_nu_cli_args.env_file.is_some() { if parsed_nu_cli_args.env_file.is_some() {
config_files::read_config_file(engine_state, &mut stack, parsed_nu_cli_args.env_file, true); config_files::read_config_file(
engine_state,
&mut stack,
parsed_nu_cli_args.env_file,
true,
);
} else { } else {
config_files::read_default_env_file(engine_state, &mut stack) config_files::read_default_env_file(engine_state, &mut stack)
} }
@ -197,6 +208,7 @@ pub(crate) fn run_file(
column!(), column!(),
use_color, use_color,
); );
}
// Regenerate the $nu constant to contain the startup time and any other potential updates // Regenerate the $nu constant to contain the startup time and any other potential updates
let nu_const = create_nu_constant(engine_state, input.span().unwrap_or_else(Span::unknown))?; let nu_const = create_nu_constant(engine_state, input.span().unwrap_or_else(Span::unknown))?;

View File

@ -243,7 +243,7 @@ fn run_in_login_mode() {
#[test] #[test]
fn run_in_not_login_mode() { fn run_in_not_login_mode() {
let child_output = std::process::Command::new(nu_test_support::fs::executable_path()) let child_output = std::process::Command::new(nu_test_support::fs::executable_path())
.args(["-c", "echo $nu.is-login"]) .args(["-n", "-c", "echo $nu.is-login"])
.output() .output()
.expect("failed to run nu"); .expect("failed to run nu");
@ -254,7 +254,7 @@ fn run_in_not_login_mode() {
#[test] #[test]
fn run_in_interactive_mode() { fn run_in_interactive_mode() {
let child_output = std::process::Command::new(nu_test_support::fs::executable_path()) let child_output = std::process::Command::new(nu_test_support::fs::executable_path())
.args(["-i", "-c", "echo $nu.is-interactive"]) .args(["-n", "-i", "-c", "echo $nu.is-interactive"])
.output() .output()
.expect("failed to run nu"); .expect("failed to run nu");
@ -265,7 +265,7 @@ fn run_in_interactive_mode() {
#[test] #[test]
fn run_in_noninteractive_mode() { fn run_in_noninteractive_mode() {
let child_output = std::process::Command::new(nu_test_support::fs::executable_path()) let child_output = std::process::Command::new(nu_test_support::fs::executable_path())
.args(["-c", "echo $nu.is-interactive"]) .args(["-n", "-c", "echo $nu.is-interactive"])
.output() .output()
.expect("failed to run nu"); .expect("failed to run nu");

View File

@ -375,7 +375,7 @@ mod nu_commands {
#[test] #[test]
fn echo_internally_externally() { fn echo_internally_externally() {
let actual = nu!(r#" let actual = nu!(r#"
nu -c "echo 'foo'" nu -n -c "echo 'foo'"
"#); "#);
assert_eq!(actual.out, "foo"); assert_eq!(actual.out, "foo");
@ -385,7 +385,7 @@ mod nu_commands {
fn failed_with_proper_exit_code() { fn failed_with_proper_exit_code() {
Playground::setup("external failed", |dirs, _sandbox| { Playground::setup("external failed", |dirs, _sandbox| {
let actual = nu!(cwd: dirs.test(), r#" let actual = nu!(cwd: dirs.test(), r#"
nu -c "cargo build | complete | get exit_code" nu -n -c "cargo build | complete | get exit_code"
"#); "#);
// cargo for non rust project's exit code is 101. // cargo for non rust project's exit code is 101.
@ -396,7 +396,7 @@ mod nu_commands {
#[test] #[test]
fn better_arg_quoting() { fn better_arg_quoting() {
let actual = nu!(r#" let actual = nu!(r#"
nu -c "\# '" nu -n -c "\# '"
"#); "#);
assert_eq!(actual.out, ""); assert_eq!(actual.out, "");
@ -405,7 +405,7 @@ mod nu_commands {
#[test] #[test]
fn command_list_arg_test() { fn command_list_arg_test() {
let actual = nu!(" let actual = nu!("
nu ...['-c' 'version'] nu ...['-n' '-c' 'version']
"); ");
assert!(actual.out.contains("version")); assert!(actual.out.contains("version"));
@ -416,7 +416,7 @@ mod nu_commands {
#[test] #[test]
fn command_cell_path_arg_test() { fn command_cell_path_arg_test() {
let actual = nu!(" let actual = nu!("
nu ...([ '-c' 'version' ]) nu ...([ '-n' '-c' 'version' ])
"); ");
assert!(actual.out.contains("version")); assert!(actual.out.contains("version"));
@ -431,7 +431,7 @@ mod nu_script {
#[test] #[test]
fn run_nu_script() { fn run_nu_script() {
let actual = nu!(cwd: "tests/fixtures/formats", " let actual = nu!(cwd: "tests/fixtures/formats", "
nu script.nu nu -n script.nu
"); ");
assert_eq!(actual.out, "done"); assert_eq!(actual.out, "done");
@ -440,7 +440,7 @@ mod nu_script {
#[test] #[test]
fn run_nu_script_multiline() { fn run_nu_script_multiline() {
let actual = nu!(cwd: "tests/fixtures/formats", " let actual = nu!(cwd: "tests/fixtures/formats", "
nu script_multiline.nu nu -n script_multiline.nu
"); ");
assert_eq!(actual.out, "23"); assert_eq!(actual.out, "23");

View File

@ -107,7 +107,7 @@ export def test [
export def "test stdlib" [ export def "test stdlib" [
--extra-args: string = '' --extra-args: string = ''
] { ] {
cargo run -- -c $" cargo run -- --no-config-file -c $"
use crates/nu-std/testing.nu use crates/nu-std/testing.nu
testing run-tests --path crates/nu-std ($extra_args) testing run-tests --path crates/nu-std ($extra_args)
" "