mirror of
https://github.com/nushell/nushell.git
synced 2025-03-29 17:16:49 +01:00
173 lines
5.0 KiB
Rust
173 lines
5.0 KiB
Rust
use crate::command_registry::CommandRegistry;
|
|
use crate::commands::classified::block::run_block;
|
|
use crate::commands::WholeStreamCommand;
|
|
use crate::prelude::*;
|
|
|
|
use futures::stream::once;
|
|
use nu_errors::ShellError;
|
|
use nu_protocol::{
|
|
hir::Block, Scope, Signature, SyntaxShape, TaggedDictBuilder, UntaggedValue, Value,
|
|
};
|
|
use nu_source::Tagged;
|
|
|
|
pub struct Each;
|
|
|
|
#[derive(Deserialize)]
|
|
pub struct EachArgs {
|
|
block: Block,
|
|
numbered: Tagged<bool>,
|
|
}
|
|
|
|
#[async_trait]
|
|
impl WholeStreamCommand for Each {
|
|
fn name(&self) -> &str {
|
|
"each"
|
|
}
|
|
|
|
fn signature(&self) -> Signature {
|
|
Signature::build("each")
|
|
.required("block", SyntaxShape::Block, "the block to run on each row")
|
|
.switch(
|
|
"numbered",
|
|
"returned a numbered item ($it.index and $it.item)",
|
|
Some('n'),
|
|
)
|
|
}
|
|
|
|
fn usage(&self) -> &str {
|
|
"Run a block on each row of the table."
|
|
}
|
|
|
|
async fn run(
|
|
&self,
|
|
args: CommandArgs,
|
|
registry: &CommandRegistry,
|
|
) -> Result<OutputStream, ShellError> {
|
|
each(args, registry).await
|
|
}
|
|
|
|
fn examples(&self) -> Vec<Example> {
|
|
vec![
|
|
Example {
|
|
description: "Echo the sum of each row",
|
|
example: "echo [[1 2] [3 4]] | each { echo $it | math sum }",
|
|
result: None,
|
|
},
|
|
Example {
|
|
description: "Echo the square of each integer",
|
|
example: "echo [1 2 3] | each { echo $(= $it * $it) }",
|
|
result: Some(vec![
|
|
UntaggedValue::int(1).into(),
|
|
UntaggedValue::int(4).into(),
|
|
UntaggedValue::int(9).into(),
|
|
]),
|
|
},
|
|
Example {
|
|
description: "Number each item and echo a message",
|
|
example:
|
|
"echo ['bob' 'fred'] | each --numbered { echo `{{$it.index}} is {{$it.item}}` }",
|
|
result: Some(vec![Value::from("0 is bob"), Value::from("1 is fred")]),
|
|
},
|
|
]
|
|
}
|
|
}
|
|
|
|
pub async fn process_row(
|
|
block: Arc<Block>,
|
|
scope: Arc<Scope>,
|
|
mut context: Arc<EvaluationContext>,
|
|
input: Value,
|
|
) -> Result<OutputStream, ShellError> {
|
|
let input_clone = input.clone();
|
|
// When we process a row, we need to know whether the block wants to have the contents of the row as
|
|
// a parameter to the block (so it gets assigned to a variable that can be used inside the block) or
|
|
// if it wants the contents as as an input stream
|
|
|
|
let input_stream = if !block.params.is_empty() {
|
|
InputStream::empty()
|
|
} else {
|
|
once(async { Ok(input_clone) }).to_input_stream()
|
|
};
|
|
|
|
let scope = if !block.params.is_empty() {
|
|
// FIXME: add check for more than parameter, once that's supported
|
|
Scope::append_var(scope, block.params[0].clone(), input)
|
|
} else {
|
|
scope
|
|
};
|
|
|
|
Ok(
|
|
run_block(&block, Arc::make_mut(&mut context), input_stream, scope)
|
|
.await?
|
|
.to_output_stream(),
|
|
)
|
|
}
|
|
|
|
pub(crate) fn make_indexed_item(index: usize, item: Value) -> Value {
|
|
let mut dict = TaggedDictBuilder::new(item.tag());
|
|
dict.insert_untagged("index", UntaggedValue::int(index));
|
|
dict.insert_value("item", item);
|
|
|
|
dict.into_value()
|
|
}
|
|
|
|
async fn each(
|
|
raw_args: CommandArgs,
|
|
registry: &CommandRegistry,
|
|
) -> Result<OutputStream, ShellError> {
|
|
let registry = registry.clone();
|
|
let scope = raw_args.call_info.scope.clone();
|
|
let context = Arc::new(EvaluationContext::from_raw(&raw_args, ®istry));
|
|
let (each_args, input): (EachArgs, _) = raw_args.process(®istry).await?;
|
|
let block = Arc::new(each_args.block);
|
|
|
|
if each_args.numbered.item {
|
|
Ok(input
|
|
.enumerate()
|
|
.then(move |input| {
|
|
let block = block.clone();
|
|
let scope = scope.clone();
|
|
let context = context.clone();
|
|
let row = make_indexed_item(input.0, input.1);
|
|
|
|
async {
|
|
match process_row(block, scope, context, row).await {
|
|
Ok(s) => s,
|
|
Err(e) => OutputStream::one(Err(e)),
|
|
}
|
|
}
|
|
})
|
|
.flatten()
|
|
.to_output_stream())
|
|
} else {
|
|
Ok(input
|
|
.then(move |input| {
|
|
let block = block.clone();
|
|
let scope = scope.clone();
|
|
let context = context.clone();
|
|
|
|
async {
|
|
match process_row(block, scope, context, input).await {
|
|
Ok(s) => s,
|
|
Err(e) => OutputStream::one(Err(e)),
|
|
}
|
|
}
|
|
})
|
|
.flatten()
|
|
.to_output_stream())
|
|
}
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
use super::Each;
|
|
use super::ShellError;
|
|
|
|
#[test]
|
|
fn examples_work_as_expected() -> Result<(), ShellError> {
|
|
use crate::examples::test as test_examples;
|
|
|
|
Ok(test_examples(Each {})?)
|
|
}
|
|
}
|