1
0
Fork 0
mirror of https://github.com/denoland/deno.git synced 2025-01-18 03:44:05 -05:00

fix(repl): improve validator to mark more code as incomplete (#17443)

Closes #17442
This commit is contained in:
David Sherret 2023-01-16 09:45:06 -05:00 committed by GitHub
parent df4d0c55c0
commit 9d3483d4eb
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23

View file

@ -235,64 +235,80 @@ impl Validator for EditorHelper {
&self, &self,
ctx: &mut ValidationContext, ctx: &mut ValidationContext,
) -> Result<ValidationResult, ReadlineError> { ) -> Result<ValidationResult, ReadlineError> {
let mut stack: Vec<Token> = Vec::new(); Ok(validate(ctx.input()))
let mut in_template = false; }
}
for item in deno_ast::lex(ctx.input(), deno_ast::MediaType::TypeScript) { fn validate(input: &str) -> ValidationResult {
if let deno_ast::TokenOrComment::Token(token) = item.inner { let line_info = text_lines::TextLines::new(input);
match token { let mut stack: Vec<Token> = Vec::new();
Token::BinOp(BinOpToken::Div) let mut in_template = false;
| Token::AssignOp(AssignOp::DivAssign) => { let mut div_token_count_on_current_line = 0;
// it's too complicated to write code to detect regular expression literals let mut last_line_index = 0;
// which are no longer tokenized, so if a `/` or `/=` happens, then we bail
return Ok(ValidationResult::Valid(None)); for item in deno_ast::lex(input, deno_ast::MediaType::TypeScript) {
let current_line_index = line_info.line_index(item.range.start);
if current_line_index != last_line_index {
div_token_count_on_current_line = 0;
last_line_index = current_line_index;
}
if let deno_ast::TokenOrComment::Token(token) = item.inner {
match token {
Token::BinOp(BinOpToken::Div)
| Token::AssignOp(AssignOp::DivAssign) => {
// it's too complicated to write code to detect regular expression literals
// which are no longer tokenized, so if a `/` or `/=` happens twice on the same
// line, then we bail
div_token_count_on_current_line += 1;
if div_token_count_on_current_line >= 2 {
return ValidationResult::Valid(None);
} }
Token::BackQuote => in_template = !in_template,
Token::LParen
| Token::LBracket
| Token::LBrace
| Token::DollarLBrace => stack.push(token),
Token::RParen | Token::RBracket | Token::RBrace => {
match (stack.pop(), token) {
(Some(Token::LParen), Token::RParen)
| (Some(Token::LBracket), Token::RBracket)
| (Some(Token::LBrace), Token::RBrace)
| (Some(Token::DollarLBrace), Token::RBrace) => {}
(Some(left), _) => {
return Ok(ValidationResult::Invalid(Some(format!(
"Mismatched pairs: {:?} is not properly closed",
left
))))
}
(None, _) => {
// While technically invalid when unpaired, it should be V8's task to output error instead.
// Thus marked as valid with no info.
return Ok(ValidationResult::Valid(None));
}
}
}
Token::Error(error) => {
match error.kind() {
// If there is unterminated template, it continues to read input.
SyntaxError::UnterminatedTpl => {}
_ => {
// If it failed parsing, it should be V8's task to output error instead.
// Thus marked as valid with no info.
return Ok(ValidationResult::Valid(None));
}
}
}
_ => {}
} }
Token::BackQuote => in_template = !in_template,
Token::LParen
| Token::LBracket
| Token::LBrace
| Token::DollarLBrace => stack.push(token),
Token::RParen | Token::RBracket | Token::RBrace => {
match (stack.pop(), token) {
(Some(Token::LParen), Token::RParen)
| (Some(Token::LBracket), Token::RBracket)
| (Some(Token::LBrace), Token::RBrace)
| (Some(Token::DollarLBrace), Token::RBrace) => {}
(Some(left), _) => {
return ValidationResult::Invalid(Some(format!(
"Mismatched pairs: {:?} is not properly closed",
left
)))
}
(None, _) => {
// While technically invalid when unpaired, it should be V8's task to output error instead.
// Thus marked as valid with no info.
return ValidationResult::Valid(None);
}
}
}
Token::Error(error) => {
match error.kind() {
// If there is unterminated template, it continues to read input.
SyntaxError::UnterminatedTpl => {}
_ => {
// If it failed parsing, it should be V8's task to output error instead.
// Thus marked as valid with no info.
return ValidationResult::Valid(None);
}
}
}
_ => {}
} }
} }
if !stack.is_empty() || in_template {
return Ok(ValidationResult::Incomplete);
}
Ok(ValidationResult::Valid(None))
} }
if !stack.is_empty() || in_template {
return ValidationResult::Incomplete;
}
ValidationResult::Valid(None)
} }
impl Highlighter for EditorHelper { impl Highlighter for EditorHelper {
@ -512,3 +528,24 @@ impl ConditionalEventHandler for TabEventHandler {
} }
} }
} }
#[cfg(test)]
mod test {
use rustyline::validate::ValidationResult;
use super::validate;
#[test]
fn validate_only_one_forward_slash_per_line() {
let code = r#"function test(arr){
if( arr.length <= 1) return arr.map(a => a / 2)
let left = test( arr.slice( 0 , arr.length/2 ) )"#;
assert!(matches!(validate(code), ValidationResult::Incomplete));
}
#[test]
fn validate_regex_looking_code() {
let code = r#"/testing/;"#;
assert!(matches!(validate(code), ValidationResult::Valid(_)));
}
}