use nu_engine::{current_dir, eval_expression_with_input, CallExt}; use nu_protocol::ast::Call; use nu_protocol::engine::{Command, EngineState, Stack}; use nu_protocol::{Category, PipelineData, Signature, SyntaxShape, Value}; #[derive(Clone)] pub struct LetEnv; impl Command for LetEnv { fn name(&self) -> &str { "let-env" } fn usage(&self) -> &str { "Create an environment variable and give it a value." } fn signature(&self) -> nu_protocol::Signature { Signature::build("let-env") .required("var_name", SyntaxShape::String, "variable name") .required( "initial_value", SyntaxShape::Keyword(b"=".to_vec(), Box::new(SyntaxShape::Expression)), "equals sign followed by value", ) .category(Category::Env) } fn run( &self, engine_state: &EngineState, stack: &mut Stack, call: &Call, input: PipelineData, ) -> Result { let env_var = call.req(engine_state, stack, 0)?; let keyword_expr = call.positional[1] .as_keyword() .expect("internal error: missing keyword"); let rhs = eval_expression_with_input(engine_state, stack, keyword_expr, input, false)? .into_value(call.head); if env_var == "PWD" { let cwd = current_dir(engine_state, stack)?; let rhs = rhs.as_string()?; let rhs = nu_path::expand_path_with(rhs, cwd); stack.add_env_var( env_var, Value::String { val: rhs.to_string_lossy().to_string(), span: call.head, }, ); } else { stack.add_env_var(env_var, rhs); } Ok(PipelineData::new(call.head)) } }