require variable names to follow additional restrictions (#6125)

This commit is contained in:
JT 2022-07-27 14:08:54 +12:00 committed by GitHub
parent d42cfab6ef
commit 9695331eed
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 133 additions and 19 deletions

View file

@ -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

View file

@ -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)),
)
} }
} }

View file

@ -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",
) )
} }

View file

@ -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(