mirror of
https://github.com/nushell/nushell
synced 2025-01-28 12:55:40 +00:00
require variable names to follow additional restrictions (#6125)
This commit is contained in:
parent
d42cfab6ef
commit
9695331eed
4 changed files with 133 additions and 19 deletions
1
crates/nu-command/src/env/let_env.rs
vendored
1
crates/nu-command/src/env/let_env.rs
vendored
|
@ -33,6 +33,7 @@ impl Command for LetEnv {
|
||||||
call: &Call,
|
call: &Call,
|
||||||
input: PipelineData,
|
input: PipelineData,
|
||||||
) -> Result<nu_protocol::PipelineData, nu_protocol::ShellError> {
|
) -> Result<nu_protocol::PipelineData, nu_protocol::ShellError> {
|
||||||
|
// TODO: find and require the crossplatform restrictions on environment names
|
||||||
let env_var = call.req(engine_state, stack, 0)?;
|
let env_var = call.req(engine_state, stack, 0)?;
|
||||||
|
|
||||||
let keyword_expr = call
|
let keyword_expr = call
|
||||||
|
|
|
@ -44,7 +44,21 @@ pub fn garbage_pipeline(spans: &[Span]) -> Pipeline {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn is_identifier_byte(b: u8) -> bool {
|
fn is_identifier_byte(b: u8) -> bool {
|
||||||
b != b'.' && b != b'[' && b != b'(' && b != b'{'
|
b != b'.'
|
||||||
|
&& b != b'['
|
||||||
|
&& b != b'('
|
||||||
|
&& b != b'{'
|
||||||
|
&& b != b'+'
|
||||||
|
&& b != b'-'
|
||||||
|
&& b != b'*'
|
||||||
|
&& b != b'^'
|
||||||
|
&& b != b'/'
|
||||||
|
&& b != b'='
|
||||||
|
&& b != b'!'
|
||||||
|
&& b != b'<'
|
||||||
|
&& b != b'>'
|
||||||
|
&& b != b'&'
|
||||||
|
&& b != b'|'
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_math_expression_like(
|
pub fn is_math_expression_like(
|
||||||
|
@ -2899,11 +2913,19 @@ pub fn parse_var_with_opt_type(
|
||||||
|
|
||||||
let ty = parse_type(working_set, type_bytes);
|
let ty = parse_type(working_set, type_bytes);
|
||||||
|
|
||||||
let id = working_set.add_variable(
|
let var_name = bytes[0..(bytes.len() - 1)].to_vec();
|
||||||
bytes[0..(bytes.len() - 1)].to_vec(),
|
|
||||||
spans[*spans_idx - 1],
|
if !is_variable(&var_name) {
|
||||||
ty.clone(),
|
return (
|
||||||
);
|
garbage(spans[*spans_idx]),
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
spans[*spans_idx],
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let id = working_set.add_variable(var_name, spans[*spans_idx - 1], ty.clone());
|
||||||
|
|
||||||
(
|
(
|
||||||
Expression {
|
Expression {
|
||||||
|
@ -2915,11 +2937,19 @@ pub fn parse_var_with_opt_type(
|
||||||
None,
|
None,
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
let id = working_set.add_variable(
|
let var_name = bytes[0..(bytes.len() - 1)].to_vec();
|
||||||
bytes[0..(bytes.len() - 1)].to_vec(),
|
|
||||||
spans[*spans_idx],
|
if !is_variable(&var_name) {
|
||||||
Type::Any,
|
return (
|
||||||
);
|
garbage(spans[*spans_idx]),
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
spans[*spans_idx],
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let id = working_set.add_variable(var_name, spans[*spans_idx], Type::Any);
|
||||||
(
|
(
|
||||||
Expression {
|
Expression {
|
||||||
expr: Expr::VarDecl(id),
|
expr: Expr::VarDecl(id),
|
||||||
|
@ -2931,8 +2961,23 @@ pub fn parse_var_with_opt_type(
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
let id =
|
let var_name = bytes;
|
||||||
working_set.add_variable(bytes, span(&spans[*spans_idx..*spans_idx + 1]), Type::Any);
|
|
||||||
|
if !is_variable(&var_name) {
|
||||||
|
return (
|
||||||
|
garbage(spans[*spans_idx]),
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
spans[*spans_idx],
|
||||||
|
)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let id = working_set.add_variable(
|
||||||
|
var_name,
|
||||||
|
span(&spans[*spans_idx..*spans_idx + 1]),
|
||||||
|
Type::Any,
|
||||||
|
);
|
||||||
|
|
||||||
(
|
(
|
||||||
Expression {
|
Expression {
|
||||||
|
@ -3130,7 +3175,23 @@ pub fn parse_signature_helper(
|
||||||
contents.split(|x| x == &b'(').map(|x| x.to_vec()).collect();
|
contents.split(|x| x == &b'(').map(|x| x.to_vec()).collect();
|
||||||
|
|
||||||
let long = String::from_utf8_lossy(&flags[0][2..]).to_string();
|
let long = String::from_utf8_lossy(&flags[0][2..]).to_string();
|
||||||
let variable_name = flags[0][2..].to_vec();
|
let mut variable_name = flags[0][2..].to_vec();
|
||||||
|
// Replace the '-' in a variable name with '_'
|
||||||
|
(0..variable_name.len()).for_each(|idx| {
|
||||||
|
if variable_name[idx] == b'-' {
|
||||||
|
variable_name[idx] = b'_';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if !is_variable(&variable_name) {
|
||||||
|
error = error.or_else(|| {
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
span,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
let var_id =
|
let var_id =
|
||||||
working_set.add_variable(variable_name, span, Type::Any);
|
working_set.add_variable(variable_name, span, Type::Any);
|
||||||
|
|
||||||
|
@ -3166,6 +3227,15 @@ pub fn parse_signature_helper(
|
||||||
let chars: Vec<char> = short_flag.chars().collect();
|
let chars: Vec<char> = short_flag.chars().collect();
|
||||||
let long = String::from_utf8_lossy(&flags[0][2..]).to_string();
|
let long = String::from_utf8_lossy(&flags[0][2..]).to_string();
|
||||||
let variable_name = flags[0][2..].to_vec();
|
let variable_name = flags[0][2..].to_vec();
|
||||||
|
if !is_variable(&variable_name) {
|
||||||
|
error = error.or_else(|| {
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
span,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
let var_id =
|
let var_id =
|
||||||
working_set.add_variable(variable_name, span, Type::Any);
|
working_set.add_variable(variable_name, span, Type::Any);
|
||||||
|
|
||||||
|
@ -3201,6 +3271,15 @@ pub fn parse_signature_helper(
|
||||||
let mut encoded_var_name = vec![0u8; 4];
|
let mut encoded_var_name = vec![0u8; 4];
|
||||||
let len = chars[0].encode_utf8(&mut encoded_var_name).len();
|
let len = chars[0].encode_utf8(&mut encoded_var_name).len();
|
||||||
let variable_name = encoded_var_name[0..len].to_vec();
|
let variable_name = encoded_var_name[0..len].to_vec();
|
||||||
|
if !is_variable(&variable_name) {
|
||||||
|
error = error.or_else(|| {
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
span,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
let var_id =
|
let var_id =
|
||||||
working_set.add_variable(variable_name, span, Type::Any);
|
working_set.add_variable(variable_name, span, Type::Any);
|
||||||
|
|
||||||
|
@ -3260,6 +3339,15 @@ pub fn parse_signature_helper(
|
||||||
let contents: Vec<_> = contents[..(contents.len() - 1)].into();
|
let contents: Vec<_> = contents[..(contents.len() - 1)].into();
|
||||||
let name = String::from_utf8_lossy(&contents).to_string();
|
let name = String::from_utf8_lossy(&contents).to_string();
|
||||||
|
|
||||||
|
if !is_variable(&contents) {
|
||||||
|
error = error.or_else(|| {
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
span,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
let var_id = working_set.add_variable(contents, span, Type::Any);
|
let var_id = working_set.add_variable(contents, span, Type::Any);
|
||||||
|
|
||||||
// Positional arg, optional
|
// Positional arg, optional
|
||||||
|
@ -3276,6 +3364,14 @@ pub fn parse_signature_helper(
|
||||||
} else if let Some(contents) = contents.strip_prefix(b"...") {
|
} else if let Some(contents) = contents.strip_prefix(b"...") {
|
||||||
let name = String::from_utf8_lossy(contents).to_string();
|
let name = String::from_utf8_lossy(contents).to_string();
|
||||||
let contents_vec: Vec<u8> = contents.to_vec();
|
let contents_vec: Vec<u8> = contents.to_vec();
|
||||||
|
if !is_variable(&contents_vec) {
|
||||||
|
error = error.or_else(|| {
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
span,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
let var_id =
|
let var_id =
|
||||||
working_set.add_variable(contents_vec, span, Type::Any);
|
working_set.add_variable(contents_vec, span, Type::Any);
|
||||||
|
@ -3291,6 +3387,15 @@ pub fn parse_signature_helper(
|
||||||
let name = String::from_utf8_lossy(contents).to_string();
|
let name = String::from_utf8_lossy(contents).to_string();
|
||||||
let contents_vec = contents.to_vec();
|
let contents_vec = contents.to_vec();
|
||||||
|
|
||||||
|
if !is_variable(&contents_vec) {
|
||||||
|
error = error.or_else(|| {
|
||||||
|
Some(ParseError::Expected(
|
||||||
|
"valid variable name".into(),
|
||||||
|
span,
|
||||||
|
))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
let var_id =
|
let var_id =
|
||||||
working_set.add_variable(contents_vec, span, Type::Any);
|
working_set.add_variable(contents_vec, span, Type::Any);
|
||||||
|
|
||||||
|
@ -4650,7 +4755,10 @@ pub fn parse_variable(
|
||||||
(None, None)
|
(None, None)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
(None, Some(ParseError::Expected("variable".into(), span)))
|
(
|
||||||
|
None,
|
||||||
|
Some(ParseError::Expected("valid variable name".into(), span)),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -67,7 +67,7 @@ fn do_rest_args() -> TestResult {
|
||||||
#[test]
|
#[test]
|
||||||
fn custom_switch1() -> TestResult {
|
fn custom_switch1() -> TestResult {
|
||||||
run_test(
|
run_test(
|
||||||
r#"def florb [ --dry-run: bool ] { if ($dry-run) { "foo" } else { "bar" } }; florb --dry-run"#,
|
r#"def florb [ --dry-run: bool ] { if ($dry_run) { "foo" } else { "bar" } }; florb --dry-run"#,
|
||||||
"foo",
|
"foo",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -75,7 +75,7 @@ fn custom_switch1() -> TestResult {
|
||||||
#[test]
|
#[test]
|
||||||
fn custom_switch2() -> TestResult {
|
fn custom_switch2() -> TestResult {
|
||||||
run_test(
|
run_test(
|
||||||
r#"def florb [ --dry-run: bool ] { if ($dry-run) { "foo" } else { "bar" } }; florb"#,
|
r#"def florb [ --dry-run: bool ] { if ($dry_run) { "foo" } else { "bar" } }; florb"#,
|
||||||
"bar",
|
"bar",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -83,7 +83,7 @@ fn custom_switch2() -> TestResult {
|
||||||
#[test]
|
#[test]
|
||||||
fn custom_switch3() -> TestResult {
|
fn custom_switch3() -> TestResult {
|
||||||
run_test(
|
run_test(
|
||||||
r#"def florb [ --dry-run ] { if ($dry-run) { "foo" } else { "bar" } }; florb --dry-run"#,
|
r#"def florb [ --dry-run ] { if ($dry_run) { "foo" } else { "bar" } }; florb --dry-run"#,
|
||||||
"foo",
|
"foo",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
@ -91,7 +91,7 @@ fn custom_switch3() -> TestResult {
|
||||||
#[test]
|
#[test]
|
||||||
fn custom_switch4() -> TestResult {
|
fn custom_switch4() -> TestResult {
|
||||||
run_test(
|
run_test(
|
||||||
r#"def florb [ --dry-run ] { if ($dry-run) { "foo" } else { "bar" } }; florb"#,
|
r#"def florb [ --dry-run ] { if ($dry_run) { "foo" } else { "bar" } }; florb"#,
|
||||||
"bar",
|
"bar",
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
|
@ -114,6 +114,11 @@ fn bad_var_name() -> TestResult {
|
||||||
fail_test(r#"let $"foo bar" = 4"#, "can't contain")
|
fail_test(r#"let $"foo bar" = 4"#, "can't contain")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn bad_var_name2() -> TestResult {
|
||||||
|
fail_test(r#"let $foo-bar = 4"#, "valid variable")
|
||||||
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn long_flag() -> TestResult {
|
fn long_flag() -> TestResult {
|
||||||
run_test(
|
run_test(
|
||||||
|
|
Loading…
Reference in a new issue