mirror of
https://github.com/nushell/nushell
synced 2025-01-01 15:58:55 +00:00
Cleanup let varname and rhs (#3507)
This commit is contained in:
parent
0886afe650
commit
55baee9a9a
3 changed files with 67 additions and 47 deletions
|
@ -4,7 +4,7 @@ use nu_engine::{FromValue, WholeStreamCommand};
|
||||||
|
|
||||||
use nu_errors::ShellError;
|
use nu_errors::ShellError;
|
||||||
use nu_protocol::{
|
use nu_protocol::{
|
||||||
hir::{CapturedBlock, ExternalRedirection, Literal},
|
hir::{CapturedBlock, ExternalRedirection},
|
||||||
Signature, SyntaxShape, TaggedDictBuilder, UntaggedValue, Value,
|
Signature, SyntaxShape, TaggedDictBuilder, UntaggedValue, Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -120,36 +120,12 @@ fn for_in(raw_args: CommandArgs) -> Result<OutputStream, ShellError> {
|
||||||
.positional
|
.positional
|
||||||
.expect("Internal error: type checker should require args");
|
.expect("Internal error: type checker should require args");
|
||||||
|
|
||||||
let mut var_name: String = match &positional[0].expr {
|
let var_name = positional[0].var_name()?;
|
||||||
nu_protocol::hir::Expression::FullColumnPath(path) => match &path.head.expr {
|
|
||||||
nu_protocol::hir::Expression::Variable(v, _) => v,
|
|
||||||
x => {
|
|
||||||
return Err(ShellError::labeled_error(
|
|
||||||
format!("Expected a variable (got {:?})", x),
|
|
||||||
"expected a variable",
|
|
||||||
positional[0].span,
|
|
||||||
))
|
|
||||||
}
|
|
||||||
},
|
|
||||||
nu_protocol::hir::Expression::Literal(Literal::String(x)) => x,
|
|
||||||
x => {
|
|
||||||
return Err(ShellError::labeled_error(
|
|
||||||
format!("Expected a variable (got {:?})", x),
|
|
||||||
"expected a variable",
|
|
||||||
positional[0].span,
|
|
||||||
))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.to_string();
|
|
||||||
|
|
||||||
let rhs = evaluate_baseline_expr(&positional[2], &context)?;
|
let rhs = evaluate_baseline_expr(&positional[2], &context)?;
|
||||||
|
|
||||||
let block: CapturedBlock =
|
let block: CapturedBlock =
|
||||||
FromValue::from_value(&evaluate_baseline_expr(&positional[3], &context)?)?;
|
FromValue::from_value(&evaluate_baseline_expr(&positional[3], &context)?)?;
|
||||||
|
|
||||||
if !var_name.starts_with('$') {
|
|
||||||
var_name = format!("${}", var_name);
|
|
||||||
}
|
|
||||||
|
|
||||||
let input = crate::commands::echo::expand_value_to_stream(rhs);
|
let input = crate::commands::echo::expand_value_to_stream(rhs);
|
||||||
let block = Arc::new(Box::new(block));
|
let block = Arc::new(Box::new(block));
|
||||||
|
|
||||||
|
|
|
@ -1,9 +1,11 @@
|
||||||
use crate::prelude::*;
|
use crate::prelude::*;
|
||||||
use nu_engine::{evaluate_baseline_expr, WholeStreamCommand};
|
use nu_engine::{evaluate_baseline_expr, FromValue, WholeStreamCommand};
|
||||||
|
|
||||||
use nu_errors::ShellError;
|
use nu_errors::ShellError;
|
||||||
use nu_protocol::{hir::CapturedBlock, hir::ClassifiedCommand, Signature, SyntaxShape};
|
use nu_protocol::{
|
||||||
use nu_source::Tagged;
|
hir::{CapturedBlock, ClassifiedCommand},
|
||||||
|
Signature, SyntaxShape, UntaggedValue,
|
||||||
|
};
|
||||||
|
|
||||||
pub struct Let;
|
pub struct Let;
|
||||||
|
|
||||||
|
@ -32,20 +34,41 @@ impl WholeStreamCommand for Let {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn examples(&self) -> Vec<Example> {
|
fn examples(&self) -> Vec<Example> {
|
||||||
vec![]
|
vec![
|
||||||
|
Example {
|
||||||
|
description: "Assign a simple value to a variable",
|
||||||
|
example: "let x = 3",
|
||||||
|
result: Some(vec![]),
|
||||||
|
},
|
||||||
|
Example {
|
||||||
|
description: "Assign the result of an expression to a variable",
|
||||||
|
example: "let result = (3 + 7); echo $result",
|
||||||
|
result: Some(vec![UntaggedValue::int(1).into()]),
|
||||||
|
},
|
||||||
|
Example {
|
||||||
|
description: "Create a variable using the full name",
|
||||||
|
example: "let $three = 3",
|
||||||
|
result: Some(vec![]),
|
||||||
|
},
|
||||||
|
]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn letcmd(args: CommandArgs) -> Result<ActionStream, ShellError> {
|
pub fn letcmd(args: CommandArgs) -> Result<ActionStream, ShellError> {
|
||||||
let tag = args.call_info.name_tag.clone();
|
|
||||||
let ctx = EvaluationContext::from_args(&args);
|
let ctx = EvaluationContext::from_args(&args);
|
||||||
let args = args.evaluate_once()?;
|
let positional = args
|
||||||
|
.call_info
|
||||||
|
.args
|
||||||
|
.positional
|
||||||
|
.expect("Internal error: type checker should require args");
|
||||||
|
|
||||||
//let (LetArgs { name, rhs, .. }, _) = args.process()?;
|
let var_name = positional[0].var_name()?;
|
||||||
let name: Tagged<String> = args.req(0)?;
|
let rhs_raw = evaluate_baseline_expr(&positional[2], &ctx)?;
|
||||||
let rhs: CapturedBlock = args.req(2)?;
|
let tag: Tag = positional[2].span.into();
|
||||||
|
|
||||||
let (expr, captured) = {
|
let rhs: CapturedBlock = FromValue::from_value(&rhs_raw)?;
|
||||||
|
|
||||||
|
let (expr, _) = {
|
||||||
if rhs.block.block.len() != 1 {
|
if rhs.block.block.len() != 1 {
|
||||||
return Err(ShellError::labeled_error(
|
return Err(ShellError::labeled_error(
|
||||||
"Expected a value",
|
"Expected a value",
|
||||||
|
@ -75,24 +98,15 @@ pub fn letcmd(args: CommandArgs) -> Result<ActionStream, ShellError> {
|
||||||
};
|
};
|
||||||
|
|
||||||
ctx.scope.enter_scope();
|
ctx.scope.enter_scope();
|
||||||
ctx.scope.add_vars(&captured.entries);
|
|
||||||
|
|
||||||
let value = evaluate_baseline_expr(&expr, &ctx);
|
let value = evaluate_baseline_expr(&expr, &ctx);
|
||||||
|
|
||||||
ctx.scope.exit_scope();
|
ctx.scope.exit_scope();
|
||||||
|
|
||||||
let value = value?;
|
let value = value?;
|
||||||
|
|
||||||
let name = if name.item.starts_with('$') {
|
|
||||||
name.item
|
|
||||||
} else {
|
|
||||||
format!("${}", name.item)
|
|
||||||
};
|
|
||||||
|
|
||||||
// Note: this is a special case for setting the context from a command
|
// Note: this is a special case for setting the context from a command
|
||||||
// In this case, if we don't set it now, we'll lose the scope that this
|
// In this case, if we don't set it now, we'll lose the scope that this
|
||||||
// variable should be set into.
|
// variable should be set into.
|
||||||
ctx.scope.add_var(name, value);
|
ctx.scope.add_var(var_name, value);
|
||||||
|
|
||||||
Ok(ActionStream::empty())
|
Ok(ActionStream::empty())
|
||||||
}
|
}
|
||||||
|
|
|
@ -681,6 +681,36 @@ impl SpannedExpression {
|
||||||
pub fn get_free_variables(&self, known_variables: &mut Vec<String>) -> Vec<String> {
|
pub fn get_free_variables(&self, known_variables: &mut Vec<String>) -> Vec<String> {
|
||||||
self.expr.get_free_variables(known_variables)
|
self.expr.get_free_variables(known_variables)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn var_name(&self) -> Result<String, ShellError> {
|
||||||
|
let var_name = match &self.expr {
|
||||||
|
Expression::FullColumnPath(path) => match &path.head.expr {
|
||||||
|
Expression::Variable(v, _) => v,
|
||||||
|
x => {
|
||||||
|
return Err(ShellError::labeled_error(
|
||||||
|
format!("Expected a variable (got {:?})", x),
|
||||||
|
"expected a variable",
|
||||||
|
self.span,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Expression::Literal(Literal::String(x)) => x,
|
||||||
|
x => {
|
||||||
|
return Err(ShellError::labeled_error(
|
||||||
|
format!("Expected a variable (got {:?})", x),
|
||||||
|
"expected a variable",
|
||||||
|
self.span,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.to_string();
|
||||||
|
|
||||||
|
if !var_name.starts_with('$') {
|
||||||
|
Ok(format!("${}", var_name))
|
||||||
|
} else {
|
||||||
|
Ok(var_name)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl std::ops::Deref for SpannedExpression {
|
impl std::ops::Deref for SpannedExpression {
|
||||||
|
|
Loading…
Reference in a new issue