2021-10-12 05:28:39 +02:00
|
|
|
use nu_engine::{eval_block, CallExt};
|
2021-09-29 20:25:05 +02:00
|
|
|
use nu_protocol::ast::Call;
|
2021-10-25 18:58:58 +02:00
|
|
|
use nu_protocol::engine::{Command, EngineState, Stack};
|
2021-10-25 06:01:02 +02:00
|
|
|
use nu_protocol::{PipelineData, Signature, SyntaxShape, Value};
|
2021-09-29 20:25:05 +02:00
|
|
|
|
2021-10-25 06:01:02 +02:00
|
|
|
#[derive(Clone)]
|
2021-09-29 20:25:05 +02:00
|
|
|
pub struct Do;
|
|
|
|
|
|
|
|
impl Command for Do {
|
|
|
|
fn name(&self) -> &str {
|
|
|
|
"do"
|
|
|
|
}
|
|
|
|
|
|
|
|
fn usage(&self) -> &str {
|
|
|
|
"Run a block"
|
|
|
|
}
|
|
|
|
|
|
|
|
fn signature(&self) -> nu_protocol::Signature {
|
2021-10-12 05:28:39 +02:00
|
|
|
Signature::build("do")
|
|
|
|
.desc(self.usage())
|
|
|
|
.required(
|
|
|
|
"block",
|
|
|
|
SyntaxShape::Block(Some(vec![])),
|
|
|
|
"the block to run",
|
|
|
|
)
|
|
|
|
.rest("rest", SyntaxShape::Any, "the parameter(s) for the block")
|
2021-09-29 20:25:05 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fn run(
|
|
|
|
&self,
|
2021-10-25 08:31:39 +02:00
|
|
|
engine_state: &EngineState,
|
|
|
|
stack: &mut Stack,
|
2021-09-29 20:25:05 +02:00
|
|
|
call: &Call,
|
2021-10-25 06:01:02 +02:00
|
|
|
input: PipelineData,
|
|
|
|
) -> Result<nu_protocol::PipelineData, nu_protocol::ShellError> {
|
2021-10-12 05:28:39 +02:00
|
|
|
let block_id = call.positional[0]
|
|
|
|
.as_block()
|
|
|
|
.expect("internal error: expected block");
|
2021-10-25 08:31:39 +02:00
|
|
|
let rest: Vec<Value> = call.rest(engine_state, stack, 1)?;
|
2021-09-29 20:25:05 +02:00
|
|
|
|
2021-10-25 08:31:39 +02:00
|
|
|
let block = engine_state.get_block(block_id);
|
2021-09-29 20:25:05 +02:00
|
|
|
|
2021-10-25 22:04:23 +02:00
|
|
|
let mut stack = stack.collect_captures(&block.captures);
|
2021-10-12 05:28:39 +02:00
|
|
|
|
|
|
|
let params: Vec<_> = block
|
|
|
|
.signature
|
|
|
|
.required_positional
|
|
|
|
.iter()
|
|
|
|
.chain(block.signature.optional_positional.iter())
|
|
|
|
.collect();
|
|
|
|
|
|
|
|
for param in params.iter().zip(&rest) {
|
|
|
|
if let Some(var_id) = param.0.var_id {
|
2021-10-25 08:31:39 +02:00
|
|
|
stack.add_var(var_id, param.1.clone())
|
2021-10-12 05:28:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(param) = &block.signature.rest_positional {
|
|
|
|
if rest.len() > params.len() {
|
|
|
|
let mut rest_items = vec![];
|
|
|
|
|
|
|
|
for r in rest.into_iter().skip(params.len()) {
|
|
|
|
rest_items.push(r);
|
|
|
|
}
|
|
|
|
|
|
|
|
let span = if let Some(rest_item) = rest_items.first() {
|
|
|
|
rest_item.span()?
|
|
|
|
} else {
|
|
|
|
call.head
|
|
|
|
};
|
|
|
|
|
2021-10-25 08:31:39 +02:00
|
|
|
stack.add_var(
|
2021-10-12 05:28:39 +02:00
|
|
|
param
|
|
|
|
.var_id
|
|
|
|
.expect("Internal error: rest positional parameter lacks var_id"),
|
|
|
|
Value::List {
|
|
|
|
vals: rest_items,
|
|
|
|
span,
|
|
|
|
},
|
|
|
|
)
|
2021-09-29 20:25:05 +02:00
|
|
|
}
|
|
|
|
}
|
2021-10-25 18:58:58 +02:00
|
|
|
eval_block(engine_state, &mut stack, block, input)
|
2021-09-29 20:25:05 +02:00
|
|
|
}
|
|
|
|
}
|