2020-04-30 04:18:24 +00:00
|
|
|
use crate::commands::classified::block::run_block;
|
2020-04-27 02:04:54 +00:00
|
|
|
use crate::commands::WholeStreamCommand;
|
2019-12-05 20:15:41 +00:00
|
|
|
use crate::context::CommandRegistry;
|
|
|
|
use crate::prelude::*;
|
|
|
|
use nu_errors::ShellError;
|
2020-04-27 02:04:54 +00:00
|
|
|
use nu_protocol::{ColumnPath, ReturnSuccess, Signature, SyntaxShape, UntaggedValue, Value};
|
2019-12-09 18:52:01 +00:00
|
|
|
use nu_value_ext::ValueExt;
|
2019-12-05 20:15:41 +00:00
|
|
|
|
2020-04-30 04:18:24 +00:00
|
|
|
use futures::stream::once;
|
2020-05-07 05:33:30 +00:00
|
|
|
pub struct Update;
|
2019-12-05 20:15:41 +00:00
|
|
|
|
2020-04-27 02:04:54 +00:00
|
|
|
#[derive(Deserialize)]
|
2020-05-07 05:33:30 +00:00
|
|
|
pub struct UpdateArgs {
|
2020-04-27 02:04:54 +00:00
|
|
|
field: ColumnPath,
|
|
|
|
replacement: Value,
|
|
|
|
}
|
|
|
|
|
2020-05-07 05:33:30 +00:00
|
|
|
impl WholeStreamCommand for Update {
|
2019-12-05 20:15:41 +00:00
|
|
|
fn name(&self) -> &str {
|
2020-05-07 05:33:30 +00:00
|
|
|
"update"
|
2019-12-05 20:15:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn signature(&self) -> Signature {
|
2020-05-07 05:33:30 +00:00
|
|
|
Signature::build("update")
|
2019-12-05 20:15:41 +00:00
|
|
|
.required(
|
2020-04-27 02:04:54 +00:00
|
|
|
"field",
|
2019-12-05 20:15:41 +00:00
|
|
|
SyntaxShape::ColumnPath,
|
2020-05-07 05:33:30 +00:00
|
|
|
"the name of the column to update",
|
2019-12-05 20:15:41 +00:00
|
|
|
)
|
|
|
|
.required(
|
2020-04-27 02:04:54 +00:00
|
|
|
"replacement value",
|
|
|
|
SyntaxShape::Any,
|
2019-12-05 20:15:41 +00:00
|
|
|
"the new value to give the cell(s)",
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn usage(&self) -> &str {
|
2020-05-07 05:33:30 +00:00
|
|
|
"Update an existing column to have a new value."
|
2019-12-05 20:15:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&self,
|
2020-04-27 02:04:54 +00:00
|
|
|
args: CommandArgs,
|
|
|
|
registry: &CommandRegistry,
|
2019-12-05 20:15:41 +00:00
|
|
|
) -> Result<OutputStream, ShellError> {
|
2020-05-07 05:33:30 +00:00
|
|
|
Ok(args.process_raw(registry, update)?.run())
|
2020-04-27 02:04:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-05-07 05:33:30 +00:00
|
|
|
fn update(
|
|
|
|
UpdateArgs { field, replacement }: UpdateArgs,
|
2020-04-30 04:18:24 +00:00
|
|
|
context: RunnableContext,
|
|
|
|
raw_args: RawCommandArgs,
|
2020-04-27 02:04:54 +00:00
|
|
|
) -> Result<OutputStream, ShellError> {
|
2020-04-30 04:18:24 +00:00
|
|
|
let scope = raw_args.call_info.scope.clone();
|
|
|
|
let registry = context.registry.clone();
|
|
|
|
let mut input_stream = context.input;
|
2019-12-05 20:15:41 +00:00
|
|
|
|
2020-04-27 02:04:54 +00:00
|
|
|
let stream = async_stream! {
|
2020-04-30 04:18:24 +00:00
|
|
|
while let Some(input) = input_stream.next().await {
|
|
|
|
let replacement = replacement.clone();
|
|
|
|
match replacement {
|
|
|
|
Value {
|
|
|
|
value: UntaggedValue::Block(block),
|
|
|
|
tag,
|
|
|
|
} => {
|
|
|
|
let mut context = Context::from_raw(&raw_args, ®istry);
|
|
|
|
let for_block = input.clone();
|
|
|
|
let input_clone = input.clone();
|
|
|
|
let input_stream = once(async { Ok(for_block) }).to_input_stream();
|
|
|
|
|
|
|
|
let result = run_block(
|
|
|
|
&block,
|
|
|
|
&mut context,
|
|
|
|
input_stream,
|
|
|
|
&scope.clone().set_it(input_clone),
|
|
|
|
).await;
|
|
|
|
|
|
|
|
match result {
|
|
|
|
Ok(mut stream) => {
|
|
|
|
let errors = context.get_errors();
|
|
|
|
if let Some(error) = errors.first() {
|
|
|
|
yield Err(error.clone());
|
|
|
|
}
|
2019-12-05 20:15:41 +00:00
|
|
|
|
2020-04-30 04:18:24 +00:00
|
|
|
match input {
|
|
|
|
obj @ Value {
|
|
|
|
value: UntaggedValue::Row(_),
|
|
|
|
..
|
|
|
|
} => {
|
|
|
|
if let Some(result) = stream.next().await {
|
|
|
|
match obj.replace_data_at_column_path(&field, result.clone()) {
|
|
|
|
Some(v) => yield Ok(ReturnSuccess::Value(v)),
|
|
|
|
None => {
|
|
|
|
yield Err(ShellError::labeled_error(
|
2020-05-07 05:33:30 +00:00
|
|
|
"update could not find place to insert column",
|
2020-04-30 04:18:24 +00:00
|
|
|
"column name",
|
|
|
|
obj.tag,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Value { tag, ..} => {
|
|
|
|
yield Err(ShellError::labeled_error(
|
|
|
|
"Unrecognized type in stream",
|
|
|
|
"original value",
|
|
|
|
tag,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Err(e) => {
|
|
|
|
yield Err(e);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
_ => {
|
|
|
|
match input {
|
|
|
|
obj @ Value {
|
|
|
|
value: UntaggedValue::Row(_),
|
|
|
|
..
|
|
|
|
} => match obj.replace_data_at_column_path(&field, replacement.clone()) {
|
|
|
|
Some(v) => yield Ok(ReturnSuccess::Value(v)),
|
|
|
|
None => {
|
|
|
|
yield Err(ShellError::labeled_error(
|
2020-05-07 05:33:30 +00:00
|
|
|
"update could not find place to insert column",
|
2020-04-30 04:18:24 +00:00
|
|
|
"column name",
|
|
|
|
obj.tag,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
},
|
|
|
|
Value { tag, ..} => {
|
|
|
|
yield Err(ShellError::labeled_error(
|
|
|
|
"Unrecognized type in stream",
|
|
|
|
"original value",
|
|
|
|
tag,
|
|
|
|
))
|
|
|
|
}
|
|
|
|
_ => {}
|
|
|
|
}
|
2019-12-05 20:15:41 +00:00
|
|
|
}
|
2020-04-30 04:18:24 +00:00
|
|
|
}}
|
2020-04-27 02:04:54 +00:00
|
|
|
};
|
2019-12-05 20:15:41 +00:00
|
|
|
|
2020-04-27 02:04:54 +00:00
|
|
|
Ok(stream.to_output_stream())
|
2019-12-05 20:15:41 +00:00
|
|
|
}
|