use crate::tests::{fail_test, run_test, run_test_contains, TestResult}; use nu_test_support::nu; use pretty_assertions::assert_eq; #[test] fn no_scope_leak1() -> TestResult { fail_test( "if false { let $x = 10 } else { let $x = 20 }; $x", "Variable not found", ) } #[test] fn no_scope_leak2() -> TestResult { fail_test( "def foo [] { $x }; def bar [] { let $x = 10; foo }; bar", "Variable not found", ) } #[test] fn no_scope_leak3() -> TestResult { run_test( "def foo [$x] { $x }; def bar [] { let $x = 10; foo 20}; bar", "20", ) } #[test] fn no_scope_leak4() -> TestResult { run_test( "def foo [$x] { $x }; def bar [] { let $x = 10; (foo 20) + $x}; bar", "30", ) } #[test] fn custom_rest_var() -> TestResult { run_test("def foo [...x] { $x.0 + $x.1 }; foo 10 80", "90") } #[test] fn def_twice_should_fail() -> TestResult { fail_test( r#"def foo [] { "foo" }; def foo [] { "bar" }"#, "defined more than once", ) } #[test] fn missing_parameters() -> TestResult { fail_test(r#"def foo {}"#, "Missing required positional") } #[test] fn flag_param_value() -> TestResult { run_test( r#"def foo [--bob: int] { $bob + 100 }; foo --bob 55"#, "155", ) } #[test] fn do_rest_args() -> TestResult { run_test(r#"(do { |...rest| $rest } 1 2).1 + 10"#, "12") } #[test] fn custom_switch1() -> TestResult { run_test( r#"def florb [ --dry-run: bool ] { if ($dry_run) { "foo" } else { "bar" } }; florb --dry-run"#, "foo", ) } #[test] fn custom_switch2() -> TestResult { run_test( r#"def florb [ --dry-run: bool ] { if ($dry_run) { "foo" } else { "bar" } }; florb"#, "bar", ) } #[test] fn custom_switch3() -> TestResult { run_test( r#"def florb [ --dry-run ] { if ($dry_run) { "foo" } else { "bar" } }; florb --dry-run"#, "foo", ) } #[test] fn custom_switch4() -> TestResult { run_test( r#"def florb [ --dry-run ] { if ($dry_run) { "foo" } else { "bar" } }; florb"#, "bar", ) } #[test] fn simple_var_closing() -> TestResult { run_test("let $x = 10; def foo [] { $x }; foo", "10") } #[test] fn predecl_check() -> TestResult { run_test("def bob [] { sam }; def sam [] { 3 }; bob", "3") } #[test] fn def_with_no_dollar() -> TestResult { run_test("def bob [x] { $x + 3 }; bob 4", "7") } #[test] fn allow_missing_optional_params() -> TestResult { run_test( "def foo [x?:int] { if $x != null { $x + 10 } else { 5 } }; foo", "5", ) } #[test] fn help_present_in_def() -> TestResult { run_test_contains( "def foo [] {}; help foo;", "Display the help message for this command", ) } #[test] fn help_not_present_in_extern() -> TestResult { run_test( "module test {export extern \"git fetch\" []}; use test `git fetch`; help git fetch | ansi strip", "Usage:\n > git fetch", ) } #[test] fn override_table() -> TestResult { run_test(r#"def table [] { "hi" }; table"#, "hi") } #[test] fn override_table_eval_file() { let actual = nu!(cwd: ".", r#"def table [] { "hi" }; table"#); assert_eq!(actual.out, "hi"); } // This test is disabled on Windows because they cause a stack overflow in CI (but not locally!). // For reasons we don't understand, the Windows CI runners are prone to stack overflow. // TODO: investigate so we can enable on Windows #[cfg(not(target_os = "windows"))] #[test] fn infinite_recursion_does_not_panic() { let actual = nu!( cwd: ".", r#" def bang [] { bang }; bang "# ); assert!(actual.err.contains("Recursion limit (50) reached")); }