forked from extern/nushell
Use Record
's public API in a bunch of places (#10927)
# Description Since #10841 the goal is to remove the implementation details of `Record` outside of core operations. To this end use Record iterators and map-like accessors in a bunch of places. In this PR I try to collect the boring cases where I don't expect any dramatic performance impacts or don't have doubts about the correctness afterwards - Use checked record construction in `nu_plugin_example` - Use `Record::into_iter` in `columns` - Use `Record` iterators in `headers` cmd - Use explicit record iterators in `split-by` - Use `Record::into_iter` in variable completions - Use `Record::values` iterator in `into sqlite` - Use `Record::iter_mut` for-loop in `default` - Change `nu_engine::nonexistent_column` to use iterator - Use `Record::columns` iter in `nu-cmd-base` - Use `Record::get_index` in `nu-command/network/http` - Use `Record.insert()` in `merge` - Refactor `move` to use encapsulated record API - Use `Record.insert()` in `explore` - Use proper `Record` API in `explore` - Remove defensiveness around record in `explore` - Use encapsulated record API in more `nu-command`s # User-Facing Changes None intentional # Tests + Formatting (-)
This commit is contained in:
parent
b03ef56bcb
commit
edbf3aaccb
@ -237,9 +237,9 @@ fn nested_suggestions(
|
|||||||
match value {
|
match value {
|
||||||
Value::Record { val, .. } => {
|
Value::Record { val, .. } => {
|
||||||
// Add all the columns as completion
|
// Add all the columns as completion
|
||||||
for item in val.cols {
|
for (col, _) in val.into_iter() {
|
||||||
output.push(Suggestion {
|
output.push(Suggestion {
|
||||||
value: item,
|
value: col,
|
||||||
description: None,
|
description: None,
|
||||||
extra: None,
|
extra: None,
|
||||||
span: current_span,
|
span: current_span,
|
||||||
|
@ -6,7 +6,7 @@ pub fn merge_descriptors(values: &[Value]) -> Vec<String> {
|
|||||||
let mut seen: IndexSet<String> = indexset! {};
|
let mut seen: IndexSet<String> = indexset! {};
|
||||||
for value in values {
|
for value in values {
|
||||||
let data_descriptors = match value {
|
let data_descriptors = match value {
|
||||||
Value::Record { val, .. } => val.cols.clone(),
|
Value::Record { val, .. } => val.columns().cloned().collect(),
|
||||||
_ => vec!["".to_string()],
|
_ => vec!["".to_string()],
|
||||||
};
|
};
|
||||||
for desc in data_descriptors {
|
for desc in data_descriptors {
|
||||||
|
@ -128,8 +128,7 @@ fn action(
|
|||||||
"({})",
|
"({})",
|
||||||
match list_value {
|
match list_value {
|
||||||
Value::Record { val, .. } => {
|
Value::Record { val, .. } => {
|
||||||
val.vals
|
val.values()
|
||||||
.iter()
|
|
||||||
.map(|rec_val| {
|
.map(|rec_val| {
|
||||||
format!("'{}'", nu_value_to_string(rec_val.clone(), ""))
|
format!("'{}'", nu_value_to_string(rec_val.clone(), ""))
|
||||||
})
|
})
|
||||||
|
@ -124,9 +124,8 @@ fn getcol(
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
Value::Record { val, .. } => Ok(val
|
Value::Record { val, .. } => Ok(val
|
||||||
.cols
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(move |x| Value::string(x, head))
|
.map(move |(x, _)| Value::string(x, head))
|
||||||
.into_pipeline_data(ctrlc)
|
.into_pipeline_data(ctrlc)
|
||||||
.set_metadata(metadata)),
|
.set_metadata(metadata)),
|
||||||
// Propagate errors
|
// Propagate errors
|
||||||
|
@ -91,17 +91,15 @@ fn default(
|
|||||||
Value::Record {
|
Value::Record {
|
||||||
val: mut record, ..
|
val: mut record, ..
|
||||||
} => {
|
} => {
|
||||||
let mut idx = 0;
|
|
||||||
let mut found = false;
|
let mut found = false;
|
||||||
|
|
||||||
while idx < record.len() {
|
for (col, val) in record.iter_mut() {
|
||||||
if record.cols[idx] == column.item {
|
if *col == column.item {
|
||||||
found = true;
|
found = true;
|
||||||
if matches!(record.vals[idx], Value::Nothing { .. }) {
|
if matches!(val, Value::Nothing { .. }) {
|
||||||
record.vals[idx] = value.clone();
|
*val = value.clone();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
idx += 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if !found {
|
if !found {
|
||||||
|
@ -314,7 +314,7 @@ fn flat_value(columns: &[CellPath], item: &Value, name_tag: Span, all: bool) ->
|
|||||||
// this can avoid output column order changed.
|
// this can avoid output column order changed.
|
||||||
if index == parent_column_index {
|
if index == parent_column_index {
|
||||||
for (col, val) in inner_cols.iter().zip(inner_vals.iter()) {
|
for (col, val) in inner_cols.iter().zip(inner_vals.iter()) {
|
||||||
if record.cols.contains(col) {
|
if record.contains(col) {
|
||||||
record.push(format!("{parent_column_name}_{col}"), val.clone());
|
record.push(format!("{parent_column_name}_{col}"), val.clone());
|
||||||
} else {
|
} else {
|
||||||
record.push(col, val.clone());
|
record.push(col, val.clone());
|
||||||
@ -329,7 +329,7 @@ fn flat_value(columns: &[CellPath], item: &Value, name_tag: Span, all: bool) ->
|
|||||||
// the flattened column may be the last column in the original table.
|
// the flattened column may be the last column in the original table.
|
||||||
if index == parent_column_index {
|
if index == parent_column_index {
|
||||||
for (col, val) in inner_cols.iter().zip(inner_vals.iter()) {
|
for (col, val) in inner_cols.iter().zip(inner_vals.iter()) {
|
||||||
if record.cols.contains(col) {
|
if record.contains(col) {
|
||||||
record.push(format!("{parent_column_name}_{col}"), val.clone());
|
record.push(format!("{parent_column_name}_{col}"), val.clone());
|
||||||
} else {
|
} else {
|
||||||
record.push(col, val.clone());
|
record.push(col, val.clone());
|
||||||
|
@ -129,7 +129,7 @@ fn extract_headers(
|
|||||||
let span = value.span();
|
let span = value.span();
|
||||||
match value {
|
match value {
|
||||||
Value::Record { val: record, .. } => {
|
Value::Record { val: record, .. } => {
|
||||||
for v in &record.vals {
|
for v in record.values() {
|
||||||
if !is_valid_header(v) {
|
if !is_valid_header(v) {
|
||||||
return Err(ShellError::TypeMismatch {
|
return Err(ShellError::TypeMismatch {
|
||||||
err_message: "needs compatible type: Null, String, Bool, Float, Int"
|
err_message: "needs compatible type: Null, String, Bool, Float, Int"
|
||||||
@ -139,10 +139,9 @@ fn extract_headers(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
let old_headers = record.cols.clone();
|
let old_headers = record.columns().cloned().collect();
|
||||||
let new_headers = record
|
let new_headers = record
|
||||||
.vals
|
.values()
|
||||||
.iter()
|
|
||||||
.enumerate()
|
.enumerate()
|
||||||
.map(|(idx, value)| {
|
.map(|(idx, value)| {
|
||||||
let col = value.into_string("", config);
|
let col = value.into_string("", config);
|
||||||
|
@ -153,20 +153,12 @@ repeating this process with row 1, and so on."#
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// TODO: rewrite to mutate the input record
|
||||||
fn do_merge(input_record: &Record, to_merge_record: &Record) -> Record {
|
fn do_merge(input_record: &Record, to_merge_record: &Record) -> Record {
|
||||||
let mut result = input_record.clone();
|
let mut result = input_record.clone();
|
||||||
|
|
||||||
for (col, val) in to_merge_record {
|
for (col, val) in to_merge_record {
|
||||||
let pos = result.cols.iter().position(|c| c == col);
|
result.insert(col, val.clone());
|
||||||
// if find, replace existing data, else, push new data.
|
|
||||||
match pos {
|
|
||||||
Some(index) => {
|
|
||||||
result.vals[index] = val.clone();
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
result.push(col, val.clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
result
|
result
|
||||||
}
|
}
|
||||||
|
@ -197,7 +197,7 @@ fn move_record_columns(
|
|||||||
// Check if before/after column exist
|
// Check if before/after column exist
|
||||||
match &before_or_after.item {
|
match &before_or_after.item {
|
||||||
BeforeOrAfter::After(after) => {
|
BeforeOrAfter::After(after) => {
|
||||||
if !record.cols.contains(after) {
|
if !record.contains(after) {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Cannot move columns".to_string(),
|
"Cannot move columns".to_string(),
|
||||||
"column does not exist".to_string(),
|
"column does not exist".to_string(),
|
||||||
@ -208,7 +208,7 @@ fn move_record_columns(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
BeforeOrAfter::Before(before) => {
|
BeforeOrAfter::Before(before) => {
|
||||||
if !record.cols.contains(before) {
|
if !record.contains(before) {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
"Cannot move columns".to_string(),
|
"Cannot move columns".to_string(),
|
||||||
"column does not exist".to_string(),
|
"column does not exist".to_string(),
|
||||||
@ -224,11 +224,7 @@ fn move_record_columns(
|
|||||||
for column in columns.iter() {
|
for column in columns.iter() {
|
||||||
let column_str = column.as_string()?;
|
let column_str = column.as_string()?;
|
||||||
|
|
||||||
if let Some(idx) = record
|
if let Some(idx) = record.index_of(&column_str) {
|
||||||
.cols
|
|
||||||
.iter()
|
|
||||||
.position(|inp_col| &column_str == inp_col)
|
|
||||||
{
|
|
||||||
column_idx.push(idx);
|
column_idx.push(idx);
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::GenericError(
|
return Err(ShellError::GenericError(
|
||||||
@ -249,7 +245,7 @@ fn move_record_columns(
|
|||||||
out.push(inp_col.clone(), inp_val.clone());
|
out.push(inp_col.clone(), inp_val.clone());
|
||||||
|
|
||||||
for idx in column_idx.iter() {
|
for idx in column_idx.iter() {
|
||||||
if let (Some(col), Some(val)) = (record.cols.get(*idx), record.vals.get(*idx)) {
|
if let Some((col, val)) = record.get_index(*idx) {
|
||||||
out.push(col.clone(), val.clone());
|
out.push(col.clone(), val.clone());
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::NushellFailedSpanned {
|
return Err(ShellError::NushellFailedSpanned {
|
||||||
@ -262,7 +258,7 @@ fn move_record_columns(
|
|||||||
}
|
}
|
||||||
BeforeOrAfter::Before(before) if before == inp_col => {
|
BeforeOrAfter::Before(before) if before == inp_col => {
|
||||||
for idx in column_idx.iter() {
|
for idx in column_idx.iter() {
|
||||||
if let (Some(col), Some(val)) = (record.cols.get(*idx), record.vals.get(*idx)) {
|
if let Some((col, val)) = record.get_index(*idx) {
|
||||||
out.push(col.clone(), val.clone());
|
out.push(col.clone(), val.clone());
|
||||||
} else {
|
} else {
|
||||||
return Err(ShellError::NushellFailedSpanned {
|
return Err(ShellError::NushellFailedSpanned {
|
||||||
|
@ -254,7 +254,7 @@ pub fn sort(
|
|||||||
) -> Result<(), ShellError> {
|
) -> Result<(), ShellError> {
|
||||||
match vec.first() {
|
match vec.first() {
|
||||||
Some(Value::Record { val, .. }) => {
|
Some(Value::Record { val, .. }) => {
|
||||||
let columns = val.cols.clone();
|
let columns: Vec<String> = val.columns().cloned().collect();
|
||||||
vec.sort_by(|a, b| process(a, b, &columns, span, insensitive, natural));
|
vec.sort_by(|a, b| process(a, b, &columns, span, insensitive, natural));
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
|
@ -185,15 +185,15 @@ pub fn data_split(
|
|||||||
let span = v.span();
|
let span = v.span();
|
||||||
match v {
|
match v {
|
||||||
Value::Record { val: grouped, .. } => {
|
Value::Record { val: grouped, .. } => {
|
||||||
for (idx, list) in grouped.vals.iter().enumerate() {
|
for (outer_key, list) in grouped.into_iter() {
|
||||||
match data_group(list, splitter, span) {
|
match data_group(&list, splitter, span) {
|
||||||
Ok(grouped_vals) => {
|
Ok(grouped_vals) => {
|
||||||
if let Value::Record { val: sub, .. } = grouped_vals {
|
if let Value::Record { val: sub, .. } = grouped_vals {
|
||||||
for (inner_idx, subset) in sub.vals.iter().enumerate() {
|
for (inner_key, subset) in sub.into_iter() {
|
||||||
let s: &mut IndexMap<String, Value> =
|
let s: &mut IndexMap<String, Value> =
|
||||||
splits.entry(sub.cols[inner_idx].clone()).or_default();
|
splits.entry(inner_key).or_default();
|
||||||
|
|
||||||
s.insert(grouped.cols[idx].clone(), subset.clone());
|
s.insert(outer_key.clone(), subset.clone());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,8 +3,8 @@ use itertools::Itertools;
|
|||||||
use nu_protocol::ast::Call;
|
use nu_protocol::ast::Call;
|
||||||
use nu_protocol::engine::{Command, EngineState, Stack};
|
use nu_protocol::engine::{Command, EngineState, Stack};
|
||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
record, Category, Example, IntoPipelineData, PipelineData, PipelineMetadata, Record,
|
record, Category, Example, IntoPipelineData, PipelineData, PipelineMetadata, ShellError,
|
||||||
ShellError, Signature, Span, Type, Value,
|
Signature, Span, Type, Value,
|
||||||
};
|
};
|
||||||
use std::collections::hash_map::IntoIter;
|
use std::collections::hash_map::IntoIter;
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
@ -190,10 +190,10 @@ fn clone_to_lowercase(value: &Value) -> Value {
|
|||||||
Value::list(vec.iter().map(clone_to_lowercase).collect(), span)
|
Value::list(vec.iter().map(clone_to_lowercase).collect(), span)
|
||||||
}
|
}
|
||||||
Value::Record { val: record, .. } => Value::record(
|
Value::Record { val: record, .. } => Value::record(
|
||||||
Record {
|
record
|
||||||
cols: record.cols.clone(),
|
.iter()
|
||||||
vals: record.vals.iter().map(clone_to_lowercase).collect(),
|
.map(|(k, v)| (k.to_owned(), clone_to_lowercase(v)))
|
||||||
},
|
.collect(),
|
||||||
span,
|
span,
|
||||||
),
|
),
|
||||||
other => other.clone(),
|
other => other.clone(),
|
||||||
@ -204,24 +204,18 @@ fn sort_attributes(val: Value) -> Value {
|
|||||||
let span = val.span();
|
let span = val.span();
|
||||||
match val {
|
match val {
|
||||||
Value::Record { val, .. } => {
|
Value::Record { val, .. } => {
|
||||||
|
// TODO: sort inplace
|
||||||
let sorted = val
|
let sorted = val
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.sorted_by(|a, b| a.0.cmp(&b.0))
|
.sorted_by(|a, b| a.0.cmp(&b.0))
|
||||||
.collect_vec();
|
.collect_vec();
|
||||||
|
|
||||||
let sorted_cols = sorted.clone().into_iter().map(|a| a.0).collect_vec();
|
let record = sorted
|
||||||
let sorted_vals = sorted
|
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|a| sort_attributes(a.1))
|
.map(|(k, v)| (k, sort_attributes(v)))
|
||||||
.collect_vec();
|
.collect();
|
||||||
|
|
||||||
Value::record(
|
Value::record(record, span)
|
||||||
Record {
|
|
||||||
cols: sorted_cols,
|
|
||||||
vals: sorted_vals,
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
Value::list(vals.into_iter().map(sort_attributes).collect_vec(), span)
|
Value::list(vals.into_iter().map(sort_attributes).collect_vec(), span)
|
||||||
|
@ -126,7 +126,7 @@ fn validate(vec: &[Value], columns: &[String], span: Span) -> Result<(), ShellEr
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(nonexistent) = nonexistent_column(columns, &record.cols) {
|
if let Some(nonexistent) = nonexistent_column(columns, record.columns()) {
|
||||||
return Err(ShellError::CantFindColumn {
|
return Err(ShellError::CantFindColumn {
|
||||||
col_name: nonexistent,
|
col_name: nonexistent,
|
||||||
span,
|
span,
|
||||||
|
@ -210,7 +210,7 @@ pub fn group_by(values: PipelineData, head: Span, config: &Config) -> (PipelineD
|
|||||||
} = val
|
} = val
|
||||||
{
|
{
|
||||||
lists
|
lists
|
||||||
.entry(record.cols.concat())
|
.entry(record.columns().map(|c| c.as_str()).collect::<String>())
|
||||||
.and_modify(|v: &mut Vec<Value>| v.push(val.clone()))
|
.and_modify(|v: &mut Vec<Value>| v.push(val.clone()))
|
||||||
.or_insert_with(|| vec![val.clone()]);
|
.or_insert_with(|| vec![val.clone()]);
|
||||||
} else {
|
} else {
|
||||||
|
@ -209,7 +209,7 @@ pub fn value_to_string(
|
|||||||
let mut row = vec![];
|
let mut row = vec![];
|
||||||
|
|
||||||
if let Value::Record { val, .. } = val {
|
if let Value::Record { val, .. } = val {
|
||||||
for val in &val.vals {
|
for val in val.values() {
|
||||||
row.push(value_to_string_without_quotes(
|
row.push(value_to_string_without_quotes(
|
||||||
val,
|
val,
|
||||||
span,
|
span,
|
||||||
|
@ -602,9 +602,12 @@ fn headers_to_nu(headers: &Headers, span: Span) -> Result<PipelineData, ShellErr
|
|||||||
for (name, values) in headers {
|
for (name, values) in headers {
|
||||||
let is_duplicate = vals.iter().any(|val| {
|
let is_duplicate = vals.iter().any(|val| {
|
||||||
if let Value::Record { val, .. } = val {
|
if let Value::Record { val, .. } = val {
|
||||||
if let Some(Value::String {
|
if let Some((
|
||||||
val: header_name, ..
|
_col,
|
||||||
}) = val.vals.get(0)
|
Value::String {
|
||||||
|
val: header_name, ..
|
||||||
|
},
|
||||||
|
)) = val.get_index(0)
|
||||||
{
|
{
|
||||||
return name == header_name;
|
return name == header_name;
|
||||||
}
|
}
|
||||||
|
@ -77,7 +77,7 @@ pub fn sort(
|
|||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
if let Some(nonexistent) = nonexistent_column(&sort_columns, &record.cols) {
|
if let Some(nonexistent) = nonexistent_column(&sort_columns, record.columns()) {
|
||||||
return Err(ShellError::CantFindColumn {
|
return Err(ShellError::CantFindColumn {
|
||||||
col_name: nonexistent,
|
col_name: nonexistent,
|
||||||
span,
|
span,
|
||||||
|
@ -3,8 +3,8 @@ use nu_engine::CallExt;
|
|||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
ast::{Call, CellPath},
|
ast::{Call, CellPath},
|
||||||
engine::{Command, EngineState, Stack},
|
engine::{Command, EngineState, Stack},
|
||||||
Category, Example, PipelineData, Record, ShellError, Signature, Span, Spanned, SyntaxShape,
|
Category, Example, PipelineData, ShellError, Signature, Span, Spanned, SyntaxShape, Type,
|
||||||
Type, Value,
|
Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
@ -181,15 +181,12 @@ fn action(input: &Value, arg: &Arguments, head: Span) -> Value {
|
|||||||
match mode {
|
match mode {
|
||||||
ActionMode::Global => match other {
|
ActionMode::Global => match other {
|
||||||
Value::Record { val: record, .. } => {
|
Value::Record { val: record, .. } => {
|
||||||
let new_vals = record.vals.iter().map(|v| action(v, arg, head)).collect();
|
let new_record = record
|
||||||
|
.iter()
|
||||||
|
.map(|(k, v)| (k.clone(), action(v, arg, head)))
|
||||||
|
.collect();
|
||||||
|
|
||||||
Value::record(
|
Value::record(new_record, span)
|
||||||
Record {
|
|
||||||
cols: record.cols.to_vec(),
|
|
||||||
vals: new_vals,
|
|
||||||
},
|
|
||||||
span,
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
let new_vals = vals.iter().map(|v| action(v, arg, head)).collect();
|
let new_vals = vals.iter().map(|v| action(v, arg, head)).collect();
|
||||||
|
@ -19,8 +19,11 @@ pub fn get_columns(input: &[Value]) -> Vec<String> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// If a column doesn't exist in the input, return it.
|
// If a column doesn't exist in the input, return it.
|
||||||
pub fn nonexistent_column(inputs: &[String], columns: &[String]) -> Option<String> {
|
pub fn nonexistent_column<'a, I>(inputs: &[String], columns: I) -> Option<String>
|
||||||
let set: HashSet<String> = HashSet::from_iter(columns.iter().cloned());
|
where
|
||||||
|
I: IntoIterator<Item = &'a String>,
|
||||||
|
{
|
||||||
|
let set: HashSet<&String> = HashSet::from_iter(columns);
|
||||||
|
|
||||||
for input in inputs {
|
for input in inputs {
|
||||||
if set.contains(input) {
|
if set.contains(input) {
|
||||||
|
@ -165,7 +165,7 @@ fn help_frame_data(
|
|||||||
|
|
||||||
let (cols, mut vals) = help_manual_data(manual, aliases);
|
let (cols, mut vals) = help_manual_data(manual, aliases);
|
||||||
let vals = vals.remove(0);
|
let vals = vals.remove(0);
|
||||||
Value::record(Record { cols, vals }, NuSpan::unknown())
|
Value::record(Record::from_raw_cols_vals(cols, vals), NuSpan::unknown())
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
let commands = Value::list(commands, NuSpan::unknown());
|
let commands = Value::list(commands, NuSpan::unknown());
|
||||||
|
@ -90,7 +90,10 @@ fn collect_external_stream(
|
|||||||
pub fn collect_input(value: Value) -> (Vec<String>, Vec<Vec<Value>>) {
|
pub fn collect_input(value: Value) -> (Vec<String>, Vec<Vec<Value>>) {
|
||||||
let span = value.span();
|
let span = value.span();
|
||||||
match value {
|
match value {
|
||||||
Value::Record { val: record, .. } => (record.cols, vec![record.vals]),
|
Value::Record { val: record, .. } => {
|
||||||
|
let (key, val) = record.into_iter().unzip();
|
||||||
|
(key, vec![val])
|
||||||
|
}
|
||||||
Value::List { vals, .. } => {
|
Value::List { vals, .. } => {
|
||||||
let mut columns = get_columns(&vals);
|
let mut columns = get_columns(&vals);
|
||||||
let data = convert_records_to_dataset(&columns, vals);
|
let data = convert_records_to_dataset(&columns, vals);
|
||||||
|
@ -1041,21 +1041,9 @@ fn set_config(hm: &mut HashMap<String, Value>, path: &[&str], value: Value) -> b
|
|||||||
match val {
|
match val {
|
||||||
Value::Record { val: record, .. } => {
|
Value::Record { val: record, .. } => {
|
||||||
if path.len() == 2 {
|
if path.len() == 2 {
|
||||||
if record.cols.len() != record.vals.len() {
|
let key = path[1];
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
let key = &path[1];
|
record.insert(key, value);
|
||||||
|
|
||||||
let pos = record.cols.iter().position(|v| v == key);
|
|
||||||
match pos {
|
|
||||||
Some(i) => {
|
|
||||||
record.vals[i] = value;
|
|
||||||
}
|
|
||||||
None => {
|
|
||||||
record.push(*key, value);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
let mut hm2: HashMap<String, Value> = HashMap::new();
|
let mut hm2: HashMap<String, Value> = HashMap::new();
|
||||||
for (k, v) in record.iter() {
|
for (k, v) in record.iter() {
|
||||||
|
@ -708,10 +708,7 @@ fn build_table_as_list(v: &RecordView) -> Value {
|
|||||||
.cloned()
|
.cloned()
|
||||||
.map(|vals| {
|
.map(|vals| {
|
||||||
Value::record(
|
Value::record(
|
||||||
Record {
|
Record::from_raw_cols_vals(headers.clone(), vals),
|
||||||
cols: headers.clone(),
|
|
||||||
vals,
|
|
||||||
},
|
|
||||||
NuSpan::unknown(),
|
NuSpan::unknown(),
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
@ -726,7 +723,7 @@ fn build_table_as_record(v: &RecordView) -> Value {
|
|||||||
let cols = layer.columns.to_vec();
|
let cols = layer.columns.to_vec();
|
||||||
let vals = layer.records.get(0).map_or(Vec::new(), |row| row.clone());
|
let vals = layer.records.get(0).map_or(Vec::new(), |row| row.clone());
|
||||||
|
|
||||||
Value::record(Record { cols, vals }, NuSpan::unknown())
|
Value::record(Record::from_raw_cols_vals(cols, vals), NuSpan::unknown())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn report_cursor_position(mode: UIMode, cursor: XYCursor) -> String {
|
fn report_cursor_position(mode: UIMode, cursor: XYCursor) -> String {
|
||||||
|
@ -67,13 +67,7 @@ impl Example {
|
|||||||
.map(|v| Value::int(v * i, call.head))
|
.map(|v| Value::int(v * i, call.head))
|
||||||
.collect::<Vec<Value>>();
|
.collect::<Vec<Value>>();
|
||||||
|
|
||||||
Value::record(
|
Value::record(Record::from_raw_cols_vals(cols.clone(), vals), call.head)
|
||||||
Record {
|
|
||||||
cols: cols.clone(),
|
|
||||||
vals,
|
|
||||||
},
|
|
||||||
call.head,
|
|
||||||
)
|
|
||||||
})
|
})
|
||||||
.collect::<Vec<Value>>();
|
.collect::<Vec<Value>>();
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user