2019-12-17 19:54:39 +01:00
|
|
|
use nu_test_support::{nu, pipeline};
|
2019-12-15 17:15:06 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn by_column() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
|
|
|
open cargo_sample.toml --raw
|
|
|
|
| lines
|
|
|
|
| skip 1
|
|
|
|
| first 4
|
2020-05-24 08:41:30 +02:00
|
|
|
| split column "="
|
2022-02-20 01:26:47 +01:00
|
|
|
| sort-by column1
|
2019-12-15 17:15:06 +01:00
|
|
|
| skip 1
|
2022-09-29 00:08:17 +02:00
|
|
|
| first
|
2022-02-20 01:26:47 +01:00
|
|
|
| get column1
|
2020-09-16 21:59:32 +02:00
|
|
|
| str trim
|
2019-12-15 17:15:06 +01:00
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2020-05-07 13:03:43 +02:00
|
|
|
assert_eq!(actual.out, "description");
|
2019-12-15 17:15:06 +01:00
|
|
|
}
|
2020-01-19 20:08:37 +01:00
|
|
|
|
2020-05-24 19:37:08 +02:00
|
|
|
#[test]
|
|
|
|
fn by_invalid_column() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
|
|
|
open cargo_sample.toml --raw
|
|
|
|
| lines
|
|
|
|
| skip 1
|
|
|
|
| first 4
|
|
|
|
| split column "="
|
|
|
|
| sort-by ColumnThatDoesNotExist
|
|
|
|
| skip 1
|
2022-09-29 00:08:17 +02:00
|
|
|
| first
|
2022-02-20 01:26:47 +01:00
|
|
|
| get column1
|
2020-09-16 21:59:32 +02:00
|
|
|
| str trim
|
2020-05-24 19:37:08 +02:00
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2022-02-22 17:32:29 +01:00
|
|
|
assert!(actual.err.contains("Cannot find column"));
|
|
|
|
assert!(actual.err.contains("value originates here"));
|
2020-05-24 19:37:08 +02:00
|
|
|
}
|
|
|
|
|
Support passing an empty list to sort, uniq, sort-by, and uniq-by (issue #5957) (#8669)
# Description
Currently, all four of these commands return a (rather-confusing)
spanless error when passed an empty list:
```
> [] | sort
Error:
× no values to work with
help: no values to work with
```
This PR changes these commands to always output `[]` if the input is
`[]`.
```
> [] | sort
╭────────────╮
│ empty list │
╰────────────╯
> [] | uniq-by foo
╭────────────╮
│ empty list │
╰────────────╯
```
I'm not sure what the original logic was here, but in the case of `sort`
and `uniq`, I think the current behavior is straightforwardly wrong.
`sort-by` and `uniq-by` are a bit more complicated, since they currently
try to perform some validation that the specified column name is present
in the input (see #8667 for problems with this validation, where a
possible outcome is removing the validation entirely). When passed `[]`,
it's not possible to do any validation because there are no records.
This opens up the possibility for situations like the following:
```
> [[foo]; [5] [6]] | where foo < 3 | sort-by bar
╭────────────╮
│ empty list │
╰────────────╯
```
I think there's a strong argument that `[]` is the best output for these
commands as well, since it makes pipelines like `$table | filter
$condition | sort-by $column` more predictable. Currently, this pipeline
will throw an error if `filter` evaluates to `[]`, but work fine
otherwise. This makes it difficult to write reliable code, especially
since users are not likely to encounter the `filter -> []` case in
testing (issue #5957). The only workaround is to insert manual checks
for an empty result. IMO, this is significantly worse than the "you can
typo a column name without getting an error" problem shown above.
Other commands that take column arguments (`get`, `select`, `rename`,
etc) already have `[] -> []`, so there's existing precedent for this
behavior.
The core question here is "what columns does `[]` have"? The current
behavior of `sort-by` is "no columns", while the current behavior of
`select` is "all possible columns". Both answers lead to accepting some
likely-buggy code without throwing on error, but in order to do better
here we would need something like `Value::Table` that tracks columns on
empty tables.
If other people disagree with this logic, I'm happy to split out the
`sort-by` and `uniq-by` changes into another PR.
# User-Facing Changes
`sort`, `uniq`, `sort-by`, and `uniq-by` now return `[]` instead of
throwing an error when input is `[]`.
# 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.
The existing behavior was not documented, and the new behavior is what
you would expect by default, so I don't think we need to update
documentation.
---------
Co-authored-by: Reilly Wood <reilly.wood@icloud.com>
2023-03-30 04:55:38 +02:00
|
|
|
#[test]
|
|
|
|
fn sort_by_empty() {
|
|
|
|
let actual = nu!("[] | sort-by foo | to nuon");
|
|
|
|
|
|
|
|
assert_eq!(actual.out, "[]");
|
|
|
|
}
|
|
|
|
|
2020-07-20 19:31:58 +02:00
|
|
|
#[test]
|
|
|
|
fn ls_sort_by_name_sensitive() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
|
|
|
open sample-ls-output.json
|
|
|
|
| sort-by name
|
|
|
|
| select name
|
2022-02-04 03:01:45 +01:00
|
|
|
| to json --raw
|
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2022-02-04 22:51:49 +01:00
|
|
|
let json_output = r#"[{"name": "B.txt"},{"name": "C"},{"name": "a.txt"}]"#;
|
2020-07-20 19:31:58 +02:00
|
|
|
|
|
|
|
assert_eq!(actual.out, json_output);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn ls_sort_by_name_insensitive() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
|
|
|
open sample-ls-output.json
|
|
|
|
| sort-by -i name
|
|
|
|
| select name
|
2022-02-04 03:01:45 +01:00
|
|
|
| to json --raw
|
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2022-02-13 03:48:50 +01:00
|
|
|
let json_output = r#"[{"name": "a.txt"},{"name": "B.txt"},{"name": "C"}]"#;
|
2020-07-20 19:31:58 +02:00
|
|
|
assert_eq!(actual.out, json_output);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn ls_sort_by_type_name_sensitive() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
|
|
|
open sample-ls-output.json
|
|
|
|
| sort-by type name
|
|
|
|
| select name type
|
2022-02-04 03:01:45 +01:00
|
|
|
| to json --raw
|
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2022-02-16 17:12:49 +01:00
|
|
|
let json_output = r#"[{"name": "C","type": "Dir"},{"name": "B.txt","type": "File"},{"name": "a.txt","type": "File"}]"#;
|
2020-07-20 19:31:58 +02:00
|
|
|
assert_eq!(actual.out, json_output);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn ls_sort_by_type_name_insensitive() {
|
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
|
|
|
open sample-ls-output.json
|
|
|
|
| sort-by -i type name
|
|
|
|
| select name type
|
2022-02-04 03:01:45 +01:00
|
|
|
| to json --raw
|
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2022-02-04 22:51:49 +01:00
|
|
|
let json_output = r#"[{"name": "C","type": "Dir"},{"name": "a.txt","type": "File"},{"name": "B.txt","type": "File"}]"#;
|
2020-07-20 19:31:58 +02:00
|
|
|
assert_eq!(actual.out, json_output);
|
|
|
|
}
|
2022-02-16 20:30:37 +01:00
|
|
|
|
|
|
|
#[test]
|
2022-11-09 23:16:51 +01:00
|
|
|
fn no_column_specified_fails() {
|
2022-02-16 20:30:37 +01:00
|
|
|
let actual = nu!(
|
|
|
|
cwd: "tests/fixtures/formats", pipeline(
|
|
|
|
r#"
|
2022-11-09 23:16:51 +01:00
|
|
|
[2 0 1] | sort-by
|
2022-02-16 20:30:37 +01:00
|
|
|
"#
|
|
|
|
));
|
|
|
|
|
2022-11-09 23:16:51 +01:00
|
|
|
assert!(actual.err.contains("missing parameter"));
|
2022-02-16 20:30:37 +01:00
|
|
|
}
|
last, skip, drop, take until, take while, skip until, skip while, where, reverse, shuffle, append, prepend and sort-by raise error when given non-lists (#7623)
Closes https://github.com/nushell/nushell/issues/6941
2022-12-31 12:35:12 +01:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn fail_on_non_iterator() {
|
|
|
|
let actual = nu!(cwd: ".", pipeline("1 | sort-by"));
|
|
|
|
|
|
|
|
assert!(actual.err.contains("only_supports_this_input_type"));
|
|
|
|
}
|