Add simple each

This commit is contained in:
JT 2021-09-03 15:45:34 +12:00
parent 750502c870
commit 6c0ce95d0f
4 changed files with 67 additions and 2 deletions

View File

@ -5,7 +5,7 @@ use nu_protocol::{
Signature, SyntaxShape, Signature, SyntaxShape,
}; };
use crate::{Alias, Benchmark, BuildString, Def, For, If, Length, Let, LetEnv}; use crate::{Alias, Benchmark, BuildString, Def, Each, For, If, Length, Let, LetEnv};
pub fn create_default_context() -> Rc<RefCell<EngineState>> { pub fn create_default_context() -> Rc<RefCell<EngineState>> {
let engine_state = Rc::new(RefCell::new(EngineState::new())); let engine_state = Rc::new(RefCell::new(EngineState::new()));
@ -31,6 +31,8 @@ pub fn create_default_context() -> Rc<RefCell<EngineState>> {
working_set.add_decl(Box::new(For)); working_set.add_decl(Box::new(For));
working_set.add_decl(Box::new(Each));
working_set.add_decl(Box::new(Benchmark)); working_set.add_decl(Box::new(Benchmark));
working_set.add_decl(Box::new(Length)); working_set.add_decl(Box::new(Length));

View File

@ -0,0 +1,61 @@
use nu_engine::{eval_block, eval_expression};
use nu_protocol::ast::Call;
use nu_protocol::engine::{Command, EvaluationContext};
use nu_protocol::{IntoValueStream, Signature, Span, SyntaxShape, Value};
pub struct Each;
impl Command for Each {
fn name(&self) -> &str {
"each"
}
fn usage(&self) -> &str {
"Run a block on each element of input"
}
fn signature(&self) -> nu_protocol::Signature {
Signature::build("each")
.required(
"var_name",
SyntaxShape::Variable,
"name of the looping variable",
)
.required("block", SyntaxShape::Block, "the block to run")
}
fn run(
&self,
context: &EvaluationContext,
call: &Call,
input: Value,
) -> Result<nu_protocol::Value, nu_protocol::ShellError> {
let var_id = call.positional[0]
.as_var()
.expect("internal error: missing variable");
let block = call.positional[1]
.as_block()
.expect("internal error: expected block");
let context = context.clone();
match input {
Value::List { val, .. } => Ok(Value::List {
val: val
.map(move |x| {
let engine_state = context.engine_state.borrow();
let block = engine_state.get_block(block);
let state = context.enter_scope();
state.add_var(var_id, x.clone());
//FIXME: DON'T UNWRAP
eval_block(&state, block, Value::nothing()).unwrap()
})
.into_value_stream(),
span: call.head,
}),
_ => Ok(Value::nothing()),
}
}
}

View File

@ -63,7 +63,7 @@ impl Command for For {
state.add_var(var_id, x.clone()); state.add_var(var_id, x.clone());
//FIXME: DON'T UNWRAP //FIXME: DON'T UNWRAP
eval_block(&state, block, input.clone()).unwrap() eval_block(&state, block, Value::nothing()).unwrap()
}) })
.into_value_stream(), .into_value_stream(),
span: call.head, span: call.head,

View File

@ -3,6 +3,7 @@ mod benchmark;
mod build_string; mod build_string;
mod def; mod def;
mod default_context; mod default_context;
mod each;
mod for_; mod for_;
mod if_; mod if_;
mod length; mod length;
@ -14,6 +15,7 @@ pub use benchmark::Benchmark;
pub use build_string::BuildString; pub use build_string::BuildString;
pub use def::Def; pub use def::Def;
pub use default_context::create_default_context; pub use default_context::create_default_context;
pub use each::Each;
pub use for_::For; pub use for_::For;
pub use if_::If; pub use if_::If;
pub use length::Length; pub use length::Length;