use crate::command_registry::CommandRegistry; use crate::commands::classified::block::run_block; use crate::commands::WholeStreamCommand; use crate::prelude::*; use nu_errors::ShellError; use nu_protocol::{ ColumnPath, Primitive, ReturnSuccess, Scope, Signature, SyntaxShape, UntaggedValue, Value, }; use nu_source::HasFallibleSpan; use nu_value_ext::ValueExt; use futures::stream::once; pub struct Update; #[derive(Deserialize)] pub struct UpdateArgs { field: ColumnPath, replacement: Value, } #[async_trait] impl WholeStreamCommand for Update { fn name(&self) -> &str { "update" } fn signature(&self) -> Signature { Signature::build("update") .required( "field", SyntaxShape::ColumnPath, "the name of the column to update", ) .required( "replacement value", SyntaxShape::Any, "the new value to give the cell(s)", ) } fn usage(&self) -> &str { "Update an existing column to have a new value." } async fn run( &self, args: CommandArgs, registry: &CommandRegistry, ) -> Result { update(args, registry).await } } async fn process_row( scope: Arc, mut context: Arc, input: Value, mut replacement: Arc, field: Arc, tag: Arc, ) -> Result { let tag = &*tag; let replacement = Arc::make_mut(&mut replacement); Ok(match replacement { Value { value: UntaggedValue::Block(block), tag: block_tag, } => { let for_block = input.clone(); let input_stream = once(async { Ok(for_block) }).to_input_stream(); let scope = Scope::append_it(scope, input.clone()); let result = run_block(&block, Arc::make_mut(&mut context), input_stream, scope).await; match result { Ok(mut stream) => { let values = stream.drain_vec().await; let errors = context.get_errors(); if let Some(error) = errors.first() { return Err(error.clone()); } let result = if values.len() == 1 { let value = values .get(0) .ok_or_else(|| ShellError::unexpected("No value to update with"))?; value.clone() } else if values.is_empty() { UntaggedValue::nothing().into_untagged_value() } else { UntaggedValue::table(&values).into_untagged_value() }; match input { obj @ Value { value: UntaggedValue::Row(_), .. } => match obj.replace_data_at_column_path(&field, result) { Some(v) => OutputStream::one(ReturnSuccess::value(v)), None => OutputStream::one(Err(ShellError::labeled_error( "update could not find place to insert column", "column name", obj.tag, ))), }, _ => OutputStream::one(Err(ShellError::labeled_error( "Unrecognized type in stream", "original value", block_tag.clone(), ))), } } Err(e) => OutputStream::one(Err(e)), } } replacement => match input { Value { value: UntaggedValue::Primitive(Primitive::Nothing), .. } => match scope .it() .unwrap_or_else(|| UntaggedValue::nothing().into_untagged_value()) .replace_data_at_column_path(&field, replacement.clone()) { Some(v) => OutputStream::one(ReturnSuccess::value(v)), None => OutputStream::one(Err(ShellError::labeled_error( "update could not find place to insert column", "column name", field.maybe_span().unwrap_or_else(|| tag.span), ))), }, Value { value: _, ref tag } => { match input.replace_data_at_column_path(&field, replacement.clone()) { Some(v) => OutputStream::one(ReturnSuccess::value(v)), None => OutputStream::one(Err(ShellError::labeled_error( "update could not find place to insert column", "column name", field.maybe_span().unwrap_or_else(|| tag.span), ))), } } }, }) } async fn update( raw_args: CommandArgs, registry: &CommandRegistry, ) -> Result { let registry = registry.clone(); let name_tag = Arc::new(raw_args.call_info.name_tag.clone()); let scope = raw_args.call_info.scope.clone(); let context = Arc::new(EvaluationContext::from_raw(&raw_args, ®istry)); let (UpdateArgs { field, replacement }, input) = raw_args.process(®istry).await?; let replacement = Arc::new(replacement); let field = Arc::new(field); Ok(input .then(move |input| { let tag = name_tag.clone(); let scope = scope.clone(); let context = context.clone(); let replacement = replacement.clone(); let field = field.clone(); async { match process_row(scope, context, input, replacement, field, tag).await { Ok(s) => s, Err(e) => OutputStream::one(Err(e)), } } }) .flatten() .to_output_stream()) } #[cfg(test)] mod tests { use super::Update; #[test] fn examples_work_as_expected() { use crate::examples::test as test_examples; test_examples(Update {}) } }