mirror of
https://github.com/nushell/nushell.git
synced 2025-04-06 01:58:16 +02:00
Fix flatten behavior (#5584)
* one step closer to flatten * integration code is passing, but still need to do one more level flatten for table * fix flatten * using match instead of several if let * make better comment * fmt code * better comment
This commit is contained in:
parent
ac30b3d108
commit
1999e0dcf3
@ -123,7 +123,20 @@ fn flatten(
|
|||||||
}
|
}
|
||||||
|
|
||||||
enum TableInside<'a> {
|
enum TableInside<'a> {
|
||||||
|
// handle for a column which contains a single list(but not list of records).
|
||||||
Entries(&'a str, &'a Span, Vec<&'a Value>),
|
Entries(&'a str, &'a Span, Vec<&'a Value>),
|
||||||
|
// handle for a column which contains a table, we can flatten the inner column to outer level
|
||||||
|
// `columns` means that for the given row, it contains `len(columns)` nested rows, and each nested row contains a list of column name.
|
||||||
|
// Likely, `values` means that for the given row, it contains `len(values)` nested rows, and each nested row contains a list of values.
|
||||||
|
//
|
||||||
|
// `parent_column_name` is handled for conflicting column name, the nested table may contains columns which has the same name
|
||||||
|
// to outer level, for that case, the output column name should be f"{parent_column_name}_{inner_column_name}".
|
||||||
|
FlatternedRows {
|
||||||
|
columns: Vec<Vec<String>>,
|
||||||
|
_span: &'a Span,
|
||||||
|
values: Vec<Vec<Value>>,
|
||||||
|
parent_column_name: &'a str,
|
||||||
|
},
|
||||||
}
|
}
|
||||||
|
|
||||||
fn flat_value(columns: &[CellPath], item: &Value, _name_tag: Span) -> Vec<Value> {
|
fn flat_value(columns: &[CellPath], item: &Value, _name_tag: Span) -> Vec<Value> {
|
||||||
@ -178,10 +191,28 @@ fn flat_value(columns: &[CellPath], item: &Value, _name_tag: Span) -> Vec<Value>
|
|||||||
cols,
|
cols,
|
||||||
vals,
|
vals,
|
||||||
span: _,
|
span: _,
|
||||||
} => cols.iter().enumerate().for_each(|(idx, column)| {
|
} => {
|
||||||
out.insert(column.to_string(), vals[idx].clone());
|
if column_requested.is_none() && !columns.is_empty() {
|
||||||
}),
|
if out.contains_key(column) {
|
||||||
|
out.insert(format!("{}_{}", column, column), value.clone());
|
||||||
|
} else {
|
||||||
|
out.insert(column.to_string(), value.clone());
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
cols.iter().enumerate().for_each(|(idx, inner_record_col)| {
|
||||||
|
if out.contains_key(inner_record_col) {
|
||||||
|
out.insert(
|
||||||
|
format!("{}_{}", column, inner_record_col),
|
||||||
|
vals[idx].clone(),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
out.insert(inner_record_col.to_string(), vals[idx].clone());
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
Value::List { vals, span: _ } if vals.iter().all(|f| f.as_record().is_ok()) => {
|
Value::List { vals, span: _ } if vals.iter().all(|f| f.as_record().is_ok()) => {
|
||||||
|
// it's a table (a list of record, we can flatten inner record)
|
||||||
let mut cs = vec![];
|
let mut cs = vec![];
|
||||||
let mut vs = vec![];
|
let mut vs = vec![];
|
||||||
|
|
||||||
@ -198,18 +229,16 @@ fn flat_value(columns: &[CellPath], item: &Value, _name_tag: Span) -> Vec<Value>
|
|||||||
} else {
|
} else {
|
||||||
out.insert(column.to_string(), value.clone());
|
out.insert(column.to_string(), value.clone());
|
||||||
}
|
}
|
||||||
continue;
|
} else {
|
||||||
}
|
let cols = cs.into_iter().map(|f| f.to_vec());
|
||||||
|
let vals = vs.into_iter().map(|f| f.to_vec());
|
||||||
|
|
||||||
let cols = cs.into_iter().flat_map(|f| f.to_vec());
|
inner_table = Some(TableInside::FlatternedRows {
|
||||||
let vals = vs.into_iter().flat_map(|f| f.to_vec());
|
columns: cols.collect(),
|
||||||
|
_span: &s,
|
||||||
for (k, v) in cols.into_iter().zip(vals.into_iter()) {
|
values: vals.collect(),
|
||||||
if out.contains_key(&k) {
|
parent_column_name: column,
|
||||||
out.insert(format!("{}_{}", column, k), v.clone());
|
});
|
||||||
} else {
|
|
||||||
out.insert(k, v.clone());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Value::List {
|
Value::List {
|
||||||
@ -265,26 +294,51 @@ fn flat_value(columns: &[CellPath], item: &Value, _name_tag: Span) -> Vec<Value>
|
|||||||
}
|
}
|
||||||
|
|
||||||
let mut expanded = vec![];
|
let mut expanded = vec![];
|
||||||
|
match inner_table {
|
||||||
|
Some(TableInside::Entries(column, _, entries)) => {
|
||||||
|
for entry in entries {
|
||||||
|
let mut base = out.clone();
|
||||||
|
|
||||||
if let Some(TableInside::Entries(column, _, entries)) = inner_table {
|
base.insert(column.to_string(), entry.clone());
|
||||||
for entry in entries {
|
let record = Value::Record {
|
||||||
let mut base = out.clone();
|
cols: base.keys().map(|f| f.to_string()).collect::<Vec<_>>(),
|
||||||
|
vals: base.values().cloned().collect(),
|
||||||
base.insert(column.to_string(), entry.clone());
|
span: tag,
|
||||||
|
};
|
||||||
|
expanded.push(record);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some(TableInside::FlatternedRows {
|
||||||
|
columns,
|
||||||
|
_span,
|
||||||
|
values,
|
||||||
|
parent_column_name,
|
||||||
|
}) => {
|
||||||
|
for (inner_cols, inner_vals) in columns.into_iter().zip(values) {
|
||||||
|
let mut base = out.clone();
|
||||||
|
for (col, val) in inner_cols.into_iter().zip(inner_vals) {
|
||||||
|
if base.contains_key(&col) {
|
||||||
|
base.insert(format!("{}_{}", parent_column_name, col), val);
|
||||||
|
} else {
|
||||||
|
base.insert(col, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
let record = Value::Record {
|
||||||
|
cols: base.keys().map(|f| f.to_string()).collect::<Vec<_>>(),
|
||||||
|
vals: base.values().cloned().collect(),
|
||||||
|
span: tag,
|
||||||
|
};
|
||||||
|
expanded.push(record);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None => {
|
||||||
let record = Value::Record {
|
let record = Value::Record {
|
||||||
cols: base.keys().map(|f| f.to_string()).collect::<Vec<_>>(),
|
cols: out.keys().map(|f| f.to_string()).collect::<Vec<_>>(),
|
||||||
vals: base.values().cloned().collect(),
|
vals: out.values().cloned().collect(),
|
||||||
span: tag,
|
span: tag,
|
||||||
};
|
};
|
||||||
expanded.push(record);
|
expanded.push(record);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
let record = Value::Record {
|
|
||||||
cols: out.keys().map(|f| f.to_string()).collect::<Vec<_>>(),
|
|
||||||
vals: out.values().cloned().collect(),
|
|
||||||
span: tag,
|
|
||||||
};
|
|
||||||
expanded.push(record);
|
|
||||||
}
|
}
|
||||||
expanded
|
expanded
|
||||||
} else if item.as_list().is_ok() {
|
} else if item.as_list().is_ok() {
|
||||||
|
@ -78,8 +78,6 @@ fn flatten_row_column_explicitly() {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: jt: needs more work
|
|
||||||
#[ignore]
|
|
||||||
#[test]
|
#[test]
|
||||||
fn flatten_row_columns_having_same_column_names_flats_separately() {
|
fn flatten_row_columns_having_same_column_names_flats_separately() {
|
||||||
Playground::setup("flatten_test_2", |dirs, sandbox| {
|
Playground::setup("flatten_test_2", |dirs, sandbox| {
|
||||||
@ -114,8 +112,6 @@ fn flatten_row_columns_having_same_column_names_flats_separately() {
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
// FIXME: jt: needs more work
|
|
||||||
#[ignore]
|
|
||||||
#[test]
|
#[test]
|
||||||
fn flatten_table_columns_explicitly() {
|
fn flatten_table_columns_explicitly() {
|
||||||
Playground::setup("flatten_test_3", |dirs, sandbox| {
|
Playground::setup("flatten_test_3", |dirs, sandbox| {
|
||||||
|
Loading…
Reference in New Issue
Block a user