use nu_engine::{CallExt, command_prelude::*, find_builtin_decl};
use nu_protocol::ir;
#[derive(Clone)]
pub struct RunInternal;
impl Command for RunInternal {
fn name(&self) -> &str {
"run-internal"
}
fn description(&self) -> &str {
"Run a built-in command by name. Used internally by `%($cmd)` dynamic dispatch."
}
fn signature(&self) -> Signature {
Signature::build("run-internal")
.input_output_types(vec![(Type::Any, Type::Any)])
.required(
"name",
SyntaxShape::String,
"The name of the built-in command to run.",
)
.rest(
"args",
SyntaxShape::Any,
"Arguments to pass to the built-in command.",
)
}
fn run(
&self,
engine_state: &EngineState,
stack: &mut Stack,
call: &Call,
input: PipelineData,
) -> Result<PipelineData, ShellError> {
let head = call.head;
let name: String = call.req(engine_state, stack, 0)?;
let decl_id = find_builtin_decl(engine_state, &name)
.ok_or(ShellError::CommandNotFound { span: head })?;
let decl = engine_state.get_decl(decl_id);
let rest_args: Vec<(Value, bool)> = call.rest_preserving_spreads(engine_state, stack, 1)?;
let mut builder = ir::Call::build(decl_id, head);
for (val, is_spread) in rest_args {
if is_spread {
match val {
Value::List { ref vals, .. } if vals.is_empty() => continue,
Value::Nothing { .. } => continue,
_ => {
builder.add_spread(stack, head, val);
}
}
} else {
builder.add_positional(stack, head, val);
}
}
builder.with(stack, |stack, engine_call| {
decl.run(engine_state, stack, engine_call, input)
})
}
}