uniq -i does not convert to lowercase (#7192) (#7209)

# Description
`uniq -i` does not convert output strings to lowercase.

Also, `uniq -i` did not ignore case in strings below the first level of
Tables and Records. Now all strings case are ignored for all children
Values for tables, Records, and List.

Fixes https://github.com/nushell/nushell/issues/7192


# Tests + Formatting
About the issue https://github.com/nushell/nushell/issues/7192, the
output will be:
```
〉[AAA BBB CCC] | uniq -i
╭───┬─────╮
│ 0 │ AAA │
│ 1 │ BBB │
│ 2 │ CCC │
╰───┴─────╯
```

About ignoring case for all children string, I expect this to be true:
```
([[origin, people];
    [World, (
        [[name, meal];
            ['Geremias', {plate: 'bitoque', carbs: 100}]
        ]
    )],
    [World, (
        [[name, meal];
            ['Martin', {plate: 'bitoque', carbs: 100}]
        ]
    )],
    [World, (
        [[name, meal];
            ['Geremias', {plate: 'Bitoque', carbs: 100}]
        ]
    )],
] | uniq -i
) == ([[origin, people];
    [World, (
        [[name, meal];
            ['Geremias', {plate: 'bitoque', carbs: 100}]
        ]
    )],
    [World, (
        [[name, meal];
            ['Martin', {plate: 'bitoque', carbs: 100}]
        ]
    )]
])
```
This commit is contained in:
raccmonteiro 2022-11-23 23:46:20 +00:00 committed by GitHub
parent c3c3481ef5
commit 651e86a3c0
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
2 changed files with 116 additions and 27 deletions

View File

@ -123,25 +123,68 @@ impl Command for Uniq {
} }
} }
fn to_lowercase(value: nu_protocol::Value) -> nu_protocol::Value { struct ValueCounter {
match value { val: Value,
Value::String { val: s, span } => Value::String { val_to_compare: Value,
val: s.to_lowercase(), count: i64,
span, }
},
other => other, impl PartialEq<Self> for ValueCounter {
fn eq(&self, other: &Self) -> bool {
self.val == other.val
} }
} }
fn generate_results_with_count(head: Span, uniq_values: Vec<(Value, i64)>) -> Vec<Value> { impl ValueCounter {
fn new(val: Value, flag_ignore_case: bool) -> Self {
ValueCounter {
val: val.clone(),
val_to_compare: if flag_ignore_case {
clone_to_lowercase(&val)
} else {
val
},
count: 1,
}
}
}
fn clone_to_lowercase(value: &Value) -> Value {
match value {
Value::String { val: s, span } => Value::String {
val: s.clone().to_lowercase(),
span: *span,
},
Value::List { vals: vec, span } => Value::List {
vals: vec
.clone()
.into_iter()
.map(|v| clone_to_lowercase(&v))
.collect(),
span: *span,
},
Value::Record { cols, vals, span } => Value::Record {
cols: cols.clone(),
vals: vals
.clone()
.into_iter()
.map(|v| clone_to_lowercase(&v))
.collect(),
span: *span,
},
other => other.clone(),
}
}
fn generate_results_with_count(head: Span, uniq_values: Vec<ValueCounter>) -> Vec<Value> {
uniq_values uniq_values
.into_iter() .into_iter()
.map(|item| Value::Record { .map(|item| Value::Record {
cols: vec!["value".to_string(), "count".to_string()], cols: vec!["value".to_string(), "count".to_string()],
vals: vec![ vals: vec![
item.0, item.val,
Value::Int { Value::Int {
val: item.1, val: item.count,
span: head, span: head,
}, },
], ],
@ -165,33 +208,30 @@ fn uniq(
let mut uniq_values = input let mut uniq_values = input
.into_iter() .into_iter()
.map(|item| { .map(|item| ValueCounter::new(item, flag_ignore_case))
if flag_ignore_case { .fold(Vec::<ValueCounter>::new(), |mut counter, item| {
to_lowercase(item) match counter
} else { .iter_mut()
item .find(|x| x.val_to_compare == item.val_to_compare)
} {
}) Some(x) => x.count += 1,
.fold(Vec::<(Value, i64)>::new(), |mut counter, item| { None => counter.push(item),
match counter.iter_mut().find(|x| x.0 == item) {
Some(x) => x.1 += 1,
None => counter.push((item, 1)),
}; };
counter counter
}); });
if flag_show_repeated { if flag_show_repeated {
uniq_values.retain(|value_count_pair| value_count_pair.1 > 1); uniq_values.retain(|value_count_pair| value_count_pair.count > 1);
} }
if flag_only_uniques { if flag_only_uniques {
uniq_values.retain(|value_count_pair| value_count_pair.1 == 1); uniq_values.retain(|value_count_pair| value_count_pair.count == 1);
} }
let result = if flag_show_count { let result = if flag_show_count {
generate_results_with_count(head, uniq_values) generate_results_with_count(head, uniq_values)
} else { } else {
uniq_values.into_iter().map(|v| v.0).collect() uniq_values.into_iter().map(|v| v.val).collect()
}; };
Ok(Value::List { Ok(Value::List {

View File

@ -232,11 +232,11 @@ fn uniq_simple_vals_strs() {
} }
#[test] #[test]
fn with_table() { fn table() {
let actual = nu!( let actual = nu!(
cwd: "tests/fixtures/formats", pipeline( cwd: "tests/fixtures/formats", pipeline(
r#" r#"
[[fruit day]; [apple monday] [apple friday] [apple monday] [pear monday] [orange tuesday]] [[fruit day]; [apple monday] [apple friday] [Apple friday] [apple monday] [pear monday] [orange tuesday]]
| uniq | uniq
"# "#
)); ));
@ -244,10 +244,59 @@ fn with_table() {
let expected = nu!( let expected = nu!(
cwd: "tests/fixtures/formats", pipeline( cwd: "tests/fixtures/formats", pipeline(
r#" r#"
echo [[fruit day]; [apple monday] [apple friday] [pear monday] [orange tuesday]] echo [[fruit day]; [apple monday] [apple friday] [Apple friday] [pear monday] [orange tuesday]]
"# "#
)); ));
print!("{}", actual.out); print!("{}", actual.out);
print!("{}", expected.out); print!("{}", expected.out);
assert_eq!(actual.out, expected.out); assert_eq!(actual.out, expected.out);
} }
#[test]
fn table_with_ignore_case() {
let actual = nu!(
cwd: "tests/fixtures/formats", pipeline(
r#"
[[origin, people];
[World, (
[[name, meal];
['Geremias', {plate: 'bitoque', carbs: 100}]
]
)],
[World, (
[[name, meal];
['Martin', {plate: 'bitoque', carbs: 100}]
]
)],
[World, (
[[name, meal];
['Geremias', {plate: 'Bitoque', carbs: 100}]
]
)],
] | uniq -i
"#
));
let expected = nu!(
cwd: "tests/fixtures/formats", pipeline(
r#"
echo [[origin, people];
[World, (
[[name, meal];
['Geremias', {plate: 'bitoque', carbs: 100}]
]
)],
[World, (
[[name, meal];
['Martin', {plate: 'bitoque', carbs: 100}]
]
)],
]
"#
));
print!("{}", actual.out);
print!("{}", expected.out);
assert_eq!(actual.out, expected.out);
assert_eq!(actual.out, expected.out);
}