2022-02-04 02:01:45 +00:00
|
|
|
use nu_test_support::{nu, pipeline};
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn checks_all_rows_are_true() {
|
2023-04-02 15:25:05 +00:00
|
|
|
let actual = nu!(pipeline(
|
2022-02-04 02:01:45 +00:00
|
|
|
r#"
|
2023-05-17 23:55:26 +00:00
|
|
|
echo [ "Andrés", "Andrés", "Andrés" ]
|
|
|
|
| all {|it| $it == "Andrés" }
|
2022-02-04 02:01:45 +00:00
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "true");
|
2021-04-03 18:40:54 +00:00
|
|
|
}
|
|
|
|
|
2021-07-29 21:12:24 +00:00
|
|
|
#[test]
|
|
|
|
fn checks_all_rows_are_false_with_param() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!(" [1, 2, 3, 4] | all { |a| $a >= 5 } ");
|
2022-02-04 02:01:45 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "false");
|
2021-07-29 21:12:24 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn checks_all_rows_are_true_with_param() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!(" [1, 2, 3, 4] | all { |a| $a < 5 } ");
|
2022-02-04 02:01:45 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "true");
|
2021-07-29 21:12:24 +00:00
|
|
|
}
|
|
|
|
|
2021-04-03 18:40:54 +00:00
|
|
|
#[test]
|
|
|
|
fn checks_all_columns_of_a_table_is_true() {
|
2023-04-02 15:25:05 +00:00
|
|
|
let actual = nu!(pipeline(
|
2023-05-17 23:55:26 +00:00
|
|
|
"
|
|
|
|
echo [
|
|
|
|
[ first_name, last_name, rusty_at, likes ];
|
|
|
|
[ Andrés, Robalino, '10/11/2013', 1 ]
|
|
|
|
[ JT, Turner, '10/12/2013', 1 ]
|
|
|
|
[ Darren, Schroeder, '10/11/2013', 1 ]
|
|
|
|
[ Yehuda, Katz, '10/11/2013', 1 ]
|
|
|
|
]
|
|
|
|
| all {|x| $x.likes > 0 }
|
|
|
|
"
|
2022-02-04 02:01:45 +00:00
|
|
|
));
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "true");
|
2021-04-03 18:40:54 +00:00
|
|
|
}
|
2022-07-24 11:28:12 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn checks_if_all_returns_error_with_invalid_command() {
|
2023-02-25 17:36:51 +00:00
|
|
|
// Using `with-env` to remove `st` possibly being an external program
|
2023-04-02 15:25:05 +00:00
|
|
|
let actual = nu!(pipeline(
|
2022-07-24 11:28:12 +00:00
|
|
|
r#"
|
2023-02-25 17:36:51 +00:00
|
|
|
with-env {PATH: ""} {
|
|
|
|
[red orange yellow green blue purple] | all {|it| ($it | st length) > 4 }
|
|
|
|
}
|
2022-07-24 11:28:12 +00:00
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
Rewrite run_external.rs (#12921)
This PR is a complete rewrite of `run_external.rs`. The main goal of the
rewrite is improving readability, but it also fixes some bugs related to
argument handling and the PATH variable (fixes
https://github.com/nushell/nushell/issues/6011).
I'll discuss some technical details to make reviewing easier.
## Argument handling
Quoting arguments for external commands is hard. Like, *really* hard.
We've had more than a dozen issues and PRs dedicated to quoting
arguments (see Appendix) but the current implementation is still buggy.
Here's a demonstration of the buggy behavior:
```nu
let foo = "'bar'"
^touch $foo # This creates a file named `bar`, but it should be `'bar'`
^touch ...[ "'bar'" ] # Same
```
I'll describe how this PR deals with argument handling.
First, we'll introduce the concept of **bare strings**. Bare strings are
**string literals** that are either **unquoted** or **quoted by
backticks** [^1]. Strings within a list literal are NOT considered bare
strings, even if they are unquoted or quoted by backticks.
When a bare string is used as an argument to external process, we need
to perform tilde-expansion, glob-expansion, and inner-quotes-removal, in
that order. "Inner-quotes-removal" means transforming from
`--option="value"` into `--option=value`.
## `.bat` files and CMD built-ins
On Windows, `.bat` files and `.cmd` files are considered executable, but
they need `CMD.exe` as the interpreter. The Rust standard library
supports running `.bat` files directly and will spawn `CMD.exe` under
the hood (see
[documentation](https://doc.rust-lang.org/std/process/index.html#windows-argument-splitting)).
However, other extensions are not supported [^2].
Nushell also supports a selected number of CMD built-ins. The problem
with CMD is that it uses a different set of quoting rules. Correctly
quoting for CMD requires using
[Command::raw_arg()](https://doc.rust-lang.org/std/os/windows/process/trait.CommandExt.html#tymethod.raw_arg)
and manually quoting CMD special characters, on top of quoting from the
Nushell side. ~~I decided that this is too complex and chose to reject
special characters in CMD built-ins instead [^3]. Hopefully this will
not affact real-world use cases.~~ I've implemented escaping that works
reasonably well.
## `which-support` feature
The `which` crate is now a hard dependency of `nu-command`, making the
`which-support` feature essentially useless. The `which` crate is
already a hard dependency of `nu-cli`, and we should consider removing
the `which-support` feature entirely.
## Appendix
Here's a list of quoting-related issues and PRs in rough chronological
order.
* https://github.com/nushell/nushell/issues/4609
* https://github.com/nushell/nushell/issues/4631
* https://github.com/nushell/nushell/issues/4601
* https://github.com/nushell/nushell/pull/5846
* https://github.com/nushell/nushell/issues/5978
* https://github.com/nushell/nushell/pull/6014
* https://github.com/nushell/nushell/issues/6154
* https://github.com/nushell/nushell/pull/6161
* https://github.com/nushell/nushell/issues/6399
* https://github.com/nushell/nushell/pull/6420
* https://github.com/nushell/nushell/pull/6426
* https://github.com/nushell/nushell/issues/6465
* https://github.com/nushell/nushell/issues/6559
* https://github.com/nushell/nushell/pull/6560
[^1]: The idea that backtick-quoted strings act like bare strings was
introduced by Kubouch and briefly mentioned in [the language
reference](https://www.nushell.sh/lang-guide/chapters/strings_and_text.html#backtick-quotes).
[^2]: The documentation also said "running .bat scripts in this way may
be removed in the future and so should not be relied upon", which is
another reason to move away from this. But again, quoting for CMD is
hard.
[^3]: If anyone wants to try, the best resource I found on the topic is
[this](https://daviddeley.com/autohotkey/parameters/parameters.htm).
2024-05-23 02:05:27 +00:00
|
|
|
assert!(
|
|
|
|
actual.err.contains("Command `st` not found") && actual.err.contains("Did you mean `ast`?")
|
|
|
|
);
|
2022-07-24 11:28:12 +00:00
|
|
|
}
|
2022-11-01 18:36:54 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn works_with_1_param_blocks() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!("[1 2 3] | all {|e| print $e | true }");
|
2022-11-01 18:36:54 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "123true");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn works_with_0_param_blocks() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!("[1 2 3] | all {|| print $in | true }");
|
2022-11-01 18:36:54 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "123true");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn early_exits_with_1_param_blocks() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!("[1 2 3] | all {|e| print $e | false }");
|
2022-11-01 18:36:54 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "1false");
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn early_exits_with_0_param_blocks() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!("[1 2 3] | all {|| print $in | false }");
|
2022-11-01 18:36:54 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "1false");
|
|
|
|
}
|
2022-11-09 22:05:15 +00:00
|
|
|
|
2022-11-21 13:35:11 +00:00
|
|
|
#[test]
|
2023-02-02 22:59:58 +00:00
|
|
|
fn all_uses_enumerate_index() {
|
2023-05-17 23:55:26 +00:00
|
|
|
let actual = nu!("[7 8 9] | enumerate | all {|el| print $el.index | true }");
|
2022-11-21 13:35:11 +00:00
|
|
|
|
|
|
|
assert_eq!(actual.out, "012true");
|
|
|
|
}
|
|
|
|
|
2022-11-09 22:05:15 +00:00
|
|
|
#[test]
|
|
|
|
fn unique_env_each_iteration() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats",
|
Restrict closure expression to be something like `{|| ...}` (#8290)
# Description
As title, closes: #7921 closes: #8273
# User-Facing Changes
when define a closure without pipe, nushell will raise error for now:
```
❯ let x = {ss ss}
Error: nu::parser::closure_missing_pipe
× Missing || inside closure
╭─[entry #2:1:1]
1 │ let x = {ss ss}
· ───┬───
· ╰── Parsing as a closure, but || is missing
╰────
help: Try add || to the beginning of closure
```
`any`, `each`, `all`, `where` command accepts closure, it forces user
input closure like `{||`, or parse error will returned.
```
❯ {major:2, minor:1, patch:4} | values | each { into string }
Error: nu::parser::closure_missing_pipe
× Missing || inside closure
╭─[entry #4:1:1]
1 │ {major:2, minor:1, patch:4} | values | each { into string }
· ───────┬───────
· ╰── Parsing as a closure, but || is missing
╰────
help: Try add || to the beginning of closure
```
`with-env`, `do`, `def`, `try` are special, they still remain the same,
although it says that it accepts a closure, but they don't need to be
written like `{||`, it's more likely a block but can capture variable
outside of scope:
```
❯ def test [input] { echo [0 1 2] | do { do { echo $input } } }; test aaa
aaa
```
Just realize that It's a big breaking change, we need to update config
and scripts...
# Tests + Formatting
Don't forget to add tests that cover your changes.
Make sure you've run and fixed any issues with these commands:
- `cargo fmt --all -- --check` to check standard code formatting (`cargo
fmt --all` applies these changes)
- `cargo clippy --workspace -- -D warnings -D clippy::unwrap_used -A
clippy::needless_collect` to check that you're using the standard code
style
- `cargo test --workspace` to check that all tests pass
# After Submitting
If your PR had any user-facing changes, update [the
documentation](https://github.com/nushell/nushell.github.io) after the
PR is merged, if necessary. This will help us keep the docs up to date.
2023-03-17 12:36:28 +00:00
|
|
|
"[1 2] | all {|| print ($env.PWD | str ends-with 'formats') | cd '/' | true } | to nuon"
|
2022-11-09 22:05:15 +00:00
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "truetruetrue");
|
|
|
|
}
|