mirror of
https://github.com/nushell/nushell.git
synced 2025-04-18 18:28:19 +02:00
Add set-env for setting environment variables (#2802)
This commit is contained in:
parent
2a483531a4
commit
058ef69da3
@ -67,6 +67,7 @@ pub fn create_default_context(interactive: bool) -> Result<EvaluationContext, Bo
|
|||||||
// Fundamentals
|
// Fundamentals
|
||||||
whole_stream_command(NuPlugin),
|
whole_stream_command(NuPlugin),
|
||||||
whole_stream_command(Set),
|
whole_stream_command(Set),
|
||||||
|
whole_stream_command(SetEnv),
|
||||||
whole_stream_command(Def),
|
whole_stream_command(Def),
|
||||||
// System/file operations
|
// System/file operations
|
||||||
whole_stream_command(Exec),
|
whole_stream_command(Exec),
|
||||||
|
@ -101,6 +101,7 @@ pub(crate) mod select;
|
|||||||
pub(crate) mod seq;
|
pub(crate) mod seq;
|
||||||
pub(crate) mod seq_dates;
|
pub(crate) mod seq_dates;
|
||||||
pub(crate) mod set;
|
pub(crate) mod set;
|
||||||
|
pub(crate) mod set_env;
|
||||||
pub(crate) mod shells;
|
pub(crate) mod shells;
|
||||||
pub(crate) mod shuffle;
|
pub(crate) mod shuffle;
|
||||||
pub(crate) mod size;
|
pub(crate) mod size;
|
||||||
@ -247,6 +248,7 @@ pub(crate) use select::Select;
|
|||||||
pub(crate) use seq::Seq;
|
pub(crate) use seq::Seq;
|
||||||
pub(crate) use seq_dates::SeqDates;
|
pub(crate) use seq_dates::SeqDates;
|
||||||
pub(crate) use set::Set;
|
pub(crate) use set::Set;
|
||||||
|
pub(crate) use set_env::SetEnv;
|
||||||
pub(crate) use shells::Shells;
|
pub(crate) use shells::Shells;
|
||||||
pub(crate) use shuffle::Shuffle;
|
pub(crate) use shuffle::Shuffle;
|
||||||
pub(crate) use size::Size;
|
pub(crate) use size::Size;
|
||||||
|
@ -184,6 +184,10 @@ pub(crate) async fn run_internal_command(
|
|||||||
context.scope.add_var(name, value);
|
context.scope.add_var(name, value);
|
||||||
InputStream::from_stream(futures::stream::iter(vec![]))
|
InputStream::from_stream(futures::stream::iter(vec![]))
|
||||||
}
|
}
|
||||||
|
CommandAction::AddEnvVariable(name, value) => {
|
||||||
|
context.scope.add_env_var(name, value);
|
||||||
|
InputStream::from_stream(futures::stream::iter(vec![]))
|
||||||
|
}
|
||||||
CommandAction::AddPlugins(path) => {
|
CommandAction::AddPlugins(path) => {
|
||||||
match crate::plugin::scan(vec![std::path::PathBuf::from(path)]) {
|
match crate::plugin::scan(vec![std::path::PathBuf::from(path)]) {
|
||||||
Ok(plugins) => {
|
Ok(plugins) => {
|
||||||
|
@ -85,10 +85,12 @@ pub async fn set(args: CommandArgs) -> Result<OutputStream, ShellError> {
|
|||||||
ctx.scope.enter_scope();
|
ctx.scope.enter_scope();
|
||||||
ctx.scope.add_vars(&captured.entries);
|
ctx.scope.add_vars(&captured.entries);
|
||||||
|
|
||||||
let value = evaluate_baseline_expr(&expr, &ctx).await?;
|
let value = evaluate_baseline_expr(&expr, &ctx).await;
|
||||||
|
|
||||||
ctx.scope.exit_scope();
|
ctx.scope.exit_scope();
|
||||||
|
|
||||||
|
let value = value?;
|
||||||
|
|
||||||
let name = if name.item.starts_with('$') {
|
let name = if name.item.starts_with('$') {
|
||||||
name.item.clone()
|
name.item.clone()
|
||||||
} else {
|
} else {
|
||||||
|
104
crates/nu-cli/src/commands/set_env.rs
Normal file
104
crates/nu-cli/src/commands/set_env.rs
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
use crate::prelude::*;
|
||||||
|
use crate::{commands::WholeStreamCommand, evaluate::evaluate_baseline_expr};
|
||||||
|
|
||||||
|
use nu_errors::ShellError;
|
||||||
|
use nu_protocol::{
|
||||||
|
hir::CapturedBlock, hir::ClassifiedCommand, CommandAction, ReturnSuccess, Signature,
|
||||||
|
SyntaxShape,
|
||||||
|
};
|
||||||
|
use nu_source::Tagged;
|
||||||
|
|
||||||
|
pub struct SetEnv;
|
||||||
|
|
||||||
|
#[derive(Deserialize)]
|
||||||
|
pub struct SetEnvArgs {
|
||||||
|
pub name: Tagged<String>,
|
||||||
|
pub equals: Tagged<String>,
|
||||||
|
pub rhs: CapturedBlock,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl WholeStreamCommand for SetEnv {
|
||||||
|
fn name(&self) -> &str {
|
||||||
|
"set-env"
|
||||||
|
}
|
||||||
|
|
||||||
|
fn signature(&self) -> Signature {
|
||||||
|
Signature::build("set-env")
|
||||||
|
.required(
|
||||||
|
"name",
|
||||||
|
SyntaxShape::String,
|
||||||
|
"the name of the environment variable",
|
||||||
|
)
|
||||||
|
.required("equals", SyntaxShape::String, "the equals sign")
|
||||||
|
.required(
|
||||||
|
"expr",
|
||||||
|
SyntaxShape::Initializer,
|
||||||
|
"the value to set the environment variable to",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
fn usage(&self) -> &str {
|
||||||
|
"Create an environment variable and set it to a value."
|
||||||
|
}
|
||||||
|
|
||||||
|
async fn run(&self, args: CommandArgs) -> Result<OutputStream, ShellError> {
|
||||||
|
set_env(args).await
|
||||||
|
}
|
||||||
|
|
||||||
|
fn examples(&self) -> Vec<Example> {
|
||||||
|
vec![]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn set_env(args: CommandArgs) -> Result<OutputStream, ShellError> {
|
||||||
|
let tag = args.call_info.name_tag.clone();
|
||||||
|
let ctx = EvaluationContext::from_args(&args);
|
||||||
|
|
||||||
|
let (SetEnvArgs { name, rhs, .. }, _) = args.process().await?;
|
||||||
|
|
||||||
|
let (expr, captured) = {
|
||||||
|
if rhs.block.block.len() != 1 {
|
||||||
|
return Err(ShellError::labeled_error(
|
||||||
|
"Expected a value",
|
||||||
|
"expected a value",
|
||||||
|
tag,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
match rhs.block.block[0].pipelines.get(0) {
|
||||||
|
Some(item) => match item.list.get(0) {
|
||||||
|
Some(ClassifiedCommand::Expr(expr)) => (expr.clone(), rhs.captured.clone()),
|
||||||
|
_ => {
|
||||||
|
return Err(ShellError::labeled_error(
|
||||||
|
"Expected a value",
|
||||||
|
"expected a value",
|
||||||
|
tag,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
None => {
|
||||||
|
return Err(ShellError::labeled_error(
|
||||||
|
"Expected a value",
|
||||||
|
"expected a value",
|
||||||
|
tag,
|
||||||
|
));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
ctx.scope.enter_scope();
|
||||||
|
ctx.scope.add_vars(&captured.entries);
|
||||||
|
|
||||||
|
let value = evaluate_baseline_expr(&expr, &ctx).await;
|
||||||
|
|
||||||
|
ctx.scope.exit_scope();
|
||||||
|
|
||||||
|
let value = value?;
|
||||||
|
let value = value.as_string()?;
|
||||||
|
|
||||||
|
let name = name.item.clone();
|
||||||
|
|
||||||
|
Ok(OutputStream::one(ReturnSuccess::action(
|
||||||
|
CommandAction::AddEnvVariable(name, value),
|
||||||
|
)))
|
||||||
|
}
|
@ -22,6 +22,8 @@ pub enum CommandAction {
|
|||||||
EnterHelpShell(Value),
|
EnterHelpShell(Value),
|
||||||
/// Add a variable into scope
|
/// Add a variable into scope
|
||||||
AddVariable(String, Value),
|
AddVariable(String, Value),
|
||||||
|
/// Add an environment variable into scope
|
||||||
|
AddEnvVariable(String, String),
|
||||||
/// Add plugins from path given
|
/// Add plugins from path given
|
||||||
AddPlugins(String),
|
AddPlugins(String),
|
||||||
/// Go to the previous shell in the shell ring buffer
|
/// Go to the previous shell in the shell ring buffer
|
||||||
@ -46,6 +48,7 @@ impl PrettyDebug for CommandAction {
|
|||||||
CommandAction::EnterValueShell(v) => b::typed("enter value shell", v.pretty()),
|
CommandAction::EnterValueShell(v) => b::typed("enter value shell", v.pretty()),
|
||||||
CommandAction::EnterHelpShell(v) => b::typed("enter help shell", v.pretty()),
|
CommandAction::EnterHelpShell(v) => b::typed("enter help shell", v.pretty()),
|
||||||
CommandAction::AddVariable(..) => b::description("add variable"),
|
CommandAction::AddVariable(..) => b::description("add variable"),
|
||||||
|
CommandAction::AddEnvVariable(..) => b::description("add environment variable"),
|
||||||
CommandAction::AddPlugins(..) => b::description("add plugins"),
|
CommandAction::AddPlugins(..) => b::description("add plugins"),
|
||||||
CommandAction::PreviousShell => b::description("previous shell"),
|
CommandAction::PreviousShell => b::description("previous shell"),
|
||||||
CommandAction::NextShell => b::description("next shell"),
|
CommandAction::NextShell => b::description("next shell"),
|
||||||
|
@ -345,7 +345,7 @@ fn run_custom_command() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn set_variables() {
|
fn set_variable() {
|
||||||
let actual = nu!(
|
let actual = nu!(
|
||||||
cwd: ".",
|
cwd: ".",
|
||||||
r#"
|
r#"
|
||||||
@ -358,6 +358,19 @@ fn set_variables() {
|
|||||||
assert_eq!(actual.out, "17");
|
assert_eq!(actual.out, "17");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn set_env_variable() {
|
||||||
|
let actual = nu!(
|
||||||
|
cwd: ".",
|
||||||
|
r#"
|
||||||
|
set-env TESTENVVAR = "hello world"
|
||||||
|
echo $nu.env.TESTENVVAR
|
||||||
|
"#
|
||||||
|
);
|
||||||
|
|
||||||
|
assert_eq!(actual.out, "hello world");
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(feature = "which")]
|
#[cfg(feature = "which")]
|
||||||
#[test]
|
#[test]
|
||||||
fn argument_invocation_reports_errors() {
|
fn argument_invocation_reports_errors() {
|
||||||
|
Loading…
Reference in New Issue
Block a user