mirror of
https://github.com/nushell/nushell
synced 2025-01-19 08:34:30 +00:00
2b076369e0
* handle duration overflow error * handle checked_add_signed result
708 lines
16 KiB
Rust
708 lines
16 KiB
Rust
use nu_test_support::fs::Stub::EmptyFile;
|
|
#[cfg(feature = "which")]
|
|
use nu_test_support::fs::Stub::FileWithContent;
|
|
use nu_test_support::fs::Stub::FileWithContentToBeTrimmed;
|
|
use nu_test_support::nu;
|
|
use nu_test_support::pipeline;
|
|
use nu_test_support::playground::Playground;
|
|
|
|
#[test]
|
|
fn takes_rows_of_nu_value_strings_and_pipes_it_to_stdin_of_external() {
|
|
Playground::setup("internal_to_external_pipe_test_1", |dirs, sandbox| {
|
|
sandbox.with_files(vec![FileWithContentToBeTrimmed(
|
|
"nu_times.csv",
|
|
r#"
|
|
name,rusty_luck,origin
|
|
Jason,1,Canada
|
|
Jonathan,1,New Zealand
|
|
Andrés,1,Ecuador
|
|
AndKitKatz,1,Estados Unidos
|
|
"#,
|
|
)]);
|
|
|
|
let actual = nu!(
|
|
cwd: dirs.test(), pipeline(
|
|
r#"
|
|
open nu_times.csv
|
|
| get origin
|
|
| ^echo $it
|
|
| nu --testbin chop
|
|
| lines
|
|
| nth 2
|
|
| echo $it
|
|
"#
|
|
));
|
|
|
|
// chop will remove the last escaped double quote from \"Estados Unidos\"
|
|
assert_eq!(actual.out, "Ecuado");
|
|
})
|
|
}
|
|
|
|
#[cfg(feature = "directories-support")]
|
|
#[cfg(feature = "which-support")]
|
|
#[test]
|
|
fn autoenv() {
|
|
use nu_test_support::fs::Stub::FileWithContent;
|
|
Playground::setup("autoenv_test", |dirs, sandbox| {
|
|
sandbox.mkdir("foo/bar");
|
|
sandbox.mkdir("bizz/buzz");
|
|
sandbox.mkdir("foob");
|
|
|
|
// Windows uses a different command to create an empty file so we need to have different content on windows.
|
|
let full_nu_env = if cfg!(target_os = "windows") {
|
|
r#"[env]
|
|
testkey = "testvalue"
|
|
|
|
[scriptvars]
|
|
myscript = "echo myval"
|
|
|
|
[scripts]
|
|
entryscripts = ["echo nul > hello.txt"]
|
|
exitscripts = ["echo nul > bye.txt"]"#
|
|
} else {
|
|
r#"[env]
|
|
testkey = "testvalue"
|
|
|
|
[scriptvars]
|
|
myscript = "echo myval"
|
|
|
|
[scripts]
|
|
entryscripts = ["touch hello.txt"]
|
|
exitscripts = ["touch bye.txt"]"#
|
|
};
|
|
|
|
sandbox.with_files(vec![
|
|
FileWithContent(".nu-env", full_nu_env),
|
|
FileWithContent(
|
|
"foo/.nu-env",
|
|
r#"[env]
|
|
overwrite_me = "set_in_foo"
|
|
fookey = "fooval" "#,
|
|
),
|
|
FileWithContent(
|
|
"foo/bar/.nu-env",
|
|
r#"[env]
|
|
overwrite_me = "set_in_bar""#,
|
|
),
|
|
FileWithContent("bizz/.nu-env", full_nu_env),
|
|
]);
|
|
|
|
//Make sure basic keys are set
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust
|
|
echo $nu.env.testkey"#
|
|
);
|
|
assert!(actual.out.ends_with("testvalue"));
|
|
|
|
// Make sure exitscripts are run in the directory they were specified.
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust
|
|
cd ..
|
|
cd autoenv_test
|
|
ls
|
|
ls | where name == "bye.txt" | get name"#
|
|
);
|
|
assert!(actual.out.contains("bye.txt"));
|
|
|
|
// Make sure entry scripts are run
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"cd ..
|
|
autoenv trust autoenv_test
|
|
cd autoenv_test
|
|
ls | where name == "hello.txt" | get name"#
|
|
);
|
|
assert!(actual.out.contains("hello.txt"));
|
|
|
|
// If inside a directory with exitscripts, entering a subdirectory should not trigger the exitscripts.
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust
|
|
cd foob
|
|
ls | where name == "bye.txt" | get name"#
|
|
);
|
|
assert!(!actual.out.contains("bye.txt"));
|
|
|
|
// Make sure entryscripts are run when re-visiting a directory
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust bizz
|
|
cd bizz
|
|
rm hello.txt
|
|
cd ..
|
|
cd bizz
|
|
ls | where name == "hello.txt" | get name"#
|
|
);
|
|
assert!(actual.out.contains("hello.txt"));
|
|
|
|
// Entryscripts should not run after changing to a subdirectory.
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust bizz
|
|
cd bizz
|
|
cd buzz
|
|
ls | where name == hello.txt | get name"#
|
|
);
|
|
assert!(!actual.out.ends_with("hello.txt"));
|
|
|
|
//Backing out of the directory should unset the keys
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"cd ..
|
|
echo $nu.env.testkey"#
|
|
);
|
|
assert!(!actual.out.ends_with("testvalue"));
|
|
|
|
// Make sure script keys are set
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"echo $nu.env.myscript"#
|
|
);
|
|
assert!(actual.out.ends_with("myval"));
|
|
|
|
//Going to sibling directory without passing parent should work.
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust foo
|
|
cd foob
|
|
cd ../foo
|
|
echo $nu.env.fookey
|
|
cd .."#
|
|
);
|
|
assert!(actual.out.ends_with("fooval"));
|
|
|
|
//Going to sibling directory should unset keys
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"cd foo
|
|
cd ../foob
|
|
echo $nu.env.fookey
|
|
cd .."#
|
|
);
|
|
assert!(!actual.out.ends_with("fooval"));
|
|
|
|
// Make sure entry scripts are run
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"ls | where name == "hello.txt" | get name"#
|
|
);
|
|
assert!(actual.out.contains("hello.txt"));
|
|
|
|
//Variables set in parent directories should be set even if you directly cd to a subdir
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust foo
|
|
cd foo/bar
|
|
autoenv trust
|
|
echo $nu.env.fookey"#
|
|
);
|
|
assert!(actual.out.ends_with("fooval"));
|
|
|
|
//Subdirectories should overwrite the values of parent directories.
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"autoenv trust foo
|
|
cd foo/bar
|
|
autoenv trust
|
|
echo $nu.env.overwrite_me"#
|
|
);
|
|
assert!(actual.out.ends_with("set_in_bar"));
|
|
|
|
//Make sure that overwritten values are restored.
|
|
//By deleting foo/.nu-env, we make sure that the value is actually restored and not just set again by autoenv when we re-visit foo.
|
|
let actual = nu!(
|
|
cwd: dirs.test(),
|
|
r#"cd foo
|
|
cd bar
|
|
rm ../.nu-env
|
|
cd ..
|
|
echo $nu.env.overwrite_me"#
|
|
);
|
|
assert!(actual.out.ends_with("set_in_foo"))
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
fn proper_it_expansion() {
|
|
Playground::setup("ls_test_1", |dirs, sandbox| {
|
|
sandbox.with_files(vec![
|
|
EmptyFile("andres.txt"),
|
|
EmptyFile("gedge.txt"),
|
|
EmptyFile("jonathan.txt"),
|
|
EmptyFile("yehuda.txt"),
|
|
]);
|
|
|
|
let actual = nu!(
|
|
cwd: dirs.test(), pipeline(
|
|
r#"
|
|
ls | sort-by name | group-by type | each { get File.name | echo $it } | to json
|
|
"#
|
|
));
|
|
|
|
assert_eq!(
|
|
actual.out,
|
|
r#"["andres.txt","gedge.txt","jonathan.txt","yehuda.txt"]"#
|
|
);
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
fn it_expansion_of_list() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo "foo" | echo [bar $it] | to json
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "[\"bar\",\"foo\"]");
|
|
}
|
|
|
|
#[test]
|
|
fn it_expansion_of_invocation() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo $(echo "4" | echo $it | str to-int )
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "4");
|
|
}
|
|
|
|
#[test]
|
|
fn invocation_properly_redirects() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo $(nu --testbin cococo "hello") | str collect
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "hello");
|
|
}
|
|
|
|
#[test]
|
|
fn argument_invocation() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo "foo" | echo $(echo $it)
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "foo");
|
|
}
|
|
|
|
#[test]
|
|
fn invocation_handles_dot() {
|
|
Playground::setup("invocation_handles_dot", |dirs, sandbox| {
|
|
sandbox.with_files(vec![FileWithContentToBeTrimmed(
|
|
"nu_times.csv",
|
|
r#"
|
|
name,rusty_luck,origin
|
|
Jason,1,Canada
|
|
Jonathan,1,New Zealand
|
|
Andrés,1,Ecuador
|
|
AndKitKatz,1,Estados Unidos
|
|
"#,
|
|
)]);
|
|
|
|
let actual = nu!(
|
|
cwd: dirs.test(), pipeline(
|
|
r#"
|
|
echo $(open nu_times.csv)
|
|
| get name
|
|
| nu --testbin chop $it
|
|
| nth 3
|
|
| echo $it
|
|
"#
|
|
));
|
|
|
|
assert_eq!(actual.out, "AndKitKat");
|
|
})
|
|
}
|
|
|
|
#[test]
|
|
fn string_interpolation_with_it() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo "foo" | echo `{{$it}}`
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "foo");
|
|
}
|
|
|
|
#[test]
|
|
fn string_interpolation_with_column() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo '{"name": "bob"}' | from json | echo `{{name}} is cool`
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "bob is cool");
|
|
}
|
|
|
|
#[test]
|
|
fn string_interpolation_with_column2() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo '{"name": "fred"}' | from json | echo `also {{name}} is cool`
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "also fred is cool");
|
|
}
|
|
|
|
#[test]
|
|
fn string_interpolation_with_column3() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo '{"name": "sally"}' | from json | echo `also {{name}}`
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "also sally");
|
|
}
|
|
|
|
#[test]
|
|
fn string_interpolation_with_it_column_path() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo '{"name": "sammie"}' | from json | echo `{{$it.name}}`
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "sammie");
|
|
}
|
|
|
|
#[cfg(feature = "which")]
|
|
#[test]
|
|
fn argument_invocation_reports_errors() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
"echo $(ferris_is_not_here.exe)"
|
|
);
|
|
|
|
assert!(actual.err.contains("Command not found"));
|
|
}
|
|
|
|
#[test]
|
|
fn can_process_one_row_from_internal_and_pipes_it_to_stdin_of_external() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"echo "nushelll" | nu --testbin chop"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "nushell");
|
|
}
|
|
|
|
#[test]
|
|
fn echoing_ranges() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo 1..3 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "6");
|
|
}
|
|
|
|
#[test]
|
|
fn echoing_exclusive_ranges() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo 1..<4 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "6");
|
|
}
|
|
|
|
#[test]
|
|
fn table_literals1() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo [[name age]; [foo 13]] | get age
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "13");
|
|
}
|
|
|
|
#[test]
|
|
fn table_literals2() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo [[name age] ; [bob 13] [sally 20]] | get age | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "33");
|
|
}
|
|
|
|
#[test]
|
|
fn list_with_commas() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo [1, 2, 3] | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "6");
|
|
}
|
|
|
|
#[test]
|
|
fn range_with_left_var() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo [[size]; [3]] | echo $it.size..10 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "52");
|
|
}
|
|
|
|
#[test]
|
|
fn range_with_right_var() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo [[size]; [30]] | echo 4..$it.size | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "459");
|
|
}
|
|
|
|
#[test]
|
|
fn range_with_open_left() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo ..30 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "465");
|
|
}
|
|
|
|
#[test]
|
|
fn exclusive_range_with_open_left() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo ..<31 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "465");
|
|
}
|
|
|
|
#[test]
|
|
fn range_with_open_right() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo 5.. | first 10 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "95");
|
|
}
|
|
|
|
#[test]
|
|
fn exclusive_range_with_open_right() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo 5..< | first 10 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "95");
|
|
}
|
|
|
|
#[test]
|
|
fn range_with_mixed_types() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo 1..10.5 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "55");
|
|
}
|
|
|
|
#[test]
|
|
fn exclusive_range_with_mixed_types() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo 1..<10.5 | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "55");
|
|
}
|
|
|
|
#[test]
|
|
fn it_expansion_of_tables() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo foo | echo [[`foo {{$it}} bar`]; [`{{$it}} foo`]] | get "foo foo bar"
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "foo foo");
|
|
}
|
|
|
|
#[test]
|
|
fn table_with_commas() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo [[name, age, height]; [JT, 42, 185] [Unknown, 99, 99]] | get age | math sum
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "141");
|
|
}
|
|
|
|
#[test]
|
|
fn duration_overflow() {
|
|
let actual = nu!(
|
|
cwd: ".", pipeline(
|
|
r#"
|
|
ls | get modified | = $it + 1000000000000000000yr
|
|
"#)
|
|
);
|
|
|
|
assert!(actual.err.contains("Duration overflow"));
|
|
}
|
|
|
|
#[test]
|
|
fn date_and_duration_overflow() {
|
|
let actual = nu!(
|
|
cwd: ".", pipeline(
|
|
r#"
|
|
ls | get modified | = $it + 1000000yr
|
|
"#)
|
|
);
|
|
|
|
// assert_eq!(actual.err, "overflow");
|
|
assert!(actual.err.contains("Duration and date addition overflow"));
|
|
}
|
|
|
|
mod parse {
|
|
use nu_test_support::nu;
|
|
|
|
/*
|
|
The debug command's signature is:
|
|
|
|
Usage:
|
|
> debug {flags}
|
|
|
|
flags:
|
|
-h, --help: Display this help message
|
|
-r, --raw: Prints the raw value representation.
|
|
*/
|
|
|
|
#[test]
|
|
fn errors_if_flag_passed_is_not_exact() {
|
|
let actual = nu!(cwd: ".", "debug -ra");
|
|
|
|
assert!(
|
|
actual.err.contains("unexpected flag"),
|
|
format!(
|
|
"error message '{}' should contain 'unexpected flag'",
|
|
actual.err
|
|
)
|
|
);
|
|
|
|
let actual = nu!(cwd: ".", "debug --rawx");
|
|
|
|
assert!(
|
|
actual.err.contains("unexpected flag"),
|
|
format!(
|
|
"error message '{}' should contain 'unexpected flag'",
|
|
actual.err
|
|
)
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn errors_if_flag_is_not_supported() {
|
|
let actual = nu!(cwd: ".", "debug --ferris");
|
|
|
|
assert!(
|
|
actual.err.contains("unexpected flag"),
|
|
format!(
|
|
"error message '{}' should contain 'unexpected flag'",
|
|
actual.err
|
|
)
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn errors_if_passed_an_unexpected_argument() {
|
|
let actual = nu!(cwd: ".", "debug ferris");
|
|
|
|
assert!(
|
|
actual.err.contains("unexpected argument"),
|
|
format!(
|
|
"error message '{}' should contain 'unexpected argument'",
|
|
actual.err
|
|
)
|
|
);
|
|
}
|
|
}
|
|
|
|
mod tilde_expansion {
|
|
use nu_test_support::nu;
|
|
|
|
#[test]
|
|
#[should_panic]
|
|
fn as_home_directory_when_passed_as_argument_and_begins_with_tilde() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo ~
|
|
"#
|
|
);
|
|
|
|
assert!(
|
|
!actual.out.contains('~'),
|
|
format!("'{}' should not contain ~", actual.out)
|
|
);
|
|
}
|
|
|
|
#[test]
|
|
fn does_not_expand_when_passed_as_argument_and_does_not_start_with_tilde() {
|
|
let actual = nu!(
|
|
cwd: ".",
|
|
r#"
|
|
echo "1~1"
|
|
"#
|
|
);
|
|
|
|
assert_eq!(actual.out, "1~1");
|
|
}
|
|
}
|