mirror of
https://github.com/denoland/deno.git
synced 2025-01-11 00:21:05 -05:00
refactor(repl): use SWC lexer to highlight and validate (#8496)
This commit is contained in:
parent
59f10b3604
commit
228ecb0acb
2 changed files with 147 additions and 103 deletions
66
cli/ast.rs
66
cli/ast.rs
|
@ -8,11 +8,13 @@ use deno_core::serde_json;
|
||||||
use deno_core::ModuleSpecifier;
|
use deno_core::ModuleSpecifier;
|
||||||
use std::error::Error;
|
use std::error::Error;
|
||||||
use std::fmt;
|
use std::fmt;
|
||||||
|
use std::ops::Range;
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::sync::RwLock;
|
use std::sync::RwLock;
|
||||||
use swc_common::chain;
|
use swc_common::chain;
|
||||||
use swc_common::comments::Comment;
|
use swc_common::comments::Comment;
|
||||||
|
use swc_common::comments::CommentKind;
|
||||||
use swc_common::comments::SingleThreadedComments;
|
use swc_common::comments::SingleThreadedComments;
|
||||||
use swc_common::errors::Diagnostic;
|
use swc_common::errors::Diagnostic;
|
||||||
use swc_common::errors::DiagnosticBuilder;
|
use swc_common::errors::DiagnosticBuilder;
|
||||||
|
@ -32,6 +34,7 @@ use swc_ecmascript::codegen::Node;
|
||||||
use swc_ecmascript::dep_graph::analyze_dependencies;
|
use swc_ecmascript::dep_graph::analyze_dependencies;
|
||||||
use swc_ecmascript::dep_graph::DependencyDescriptor;
|
use swc_ecmascript::dep_graph::DependencyDescriptor;
|
||||||
use swc_ecmascript::parser::lexer::Lexer;
|
use swc_ecmascript::parser::lexer::Lexer;
|
||||||
|
use swc_ecmascript::parser::token::Token;
|
||||||
use swc_ecmascript::parser::EsConfig;
|
use swc_ecmascript::parser::EsConfig;
|
||||||
use swc_ecmascript::parser::JscTarget;
|
use swc_ecmascript::parser::JscTarget;
|
||||||
use swc_ecmascript::parser::StringInput;
|
use swc_ecmascript::parser::StringInput;
|
||||||
|
@ -407,6 +410,69 @@ pub fn parse(
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub enum TokenOrComment {
|
||||||
|
Token(Token),
|
||||||
|
Comment { kind: CommentKind, text: String },
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct LexedItem {
|
||||||
|
pub span: Span,
|
||||||
|
pub inner: TokenOrComment,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LexedItem {
|
||||||
|
pub fn span_as_range(&self) -> Range<usize> {
|
||||||
|
self.span.lo.0 as usize..self.span.hi.0 as usize
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn flatten_comments(
|
||||||
|
comments: SingleThreadedComments,
|
||||||
|
) -> impl Iterator<Item = Comment> {
|
||||||
|
let (leading, trailing) = comments.take_all();
|
||||||
|
let mut comments = (*leading).clone().into_inner();
|
||||||
|
comments.extend((*trailing).clone().into_inner());
|
||||||
|
comments.into_iter().flat_map(|el| el.1)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn lex(
|
||||||
|
specifier: &str,
|
||||||
|
source: &str,
|
||||||
|
media_type: &MediaType,
|
||||||
|
) -> Vec<LexedItem> {
|
||||||
|
let source_map = SourceMap::default();
|
||||||
|
let source_file = source_map.new_source_file(
|
||||||
|
FileName::Custom(specifier.to_string()),
|
||||||
|
source.to_string(),
|
||||||
|
);
|
||||||
|
let comments = SingleThreadedComments::default();
|
||||||
|
let lexer = Lexer::new(
|
||||||
|
get_syntax(media_type),
|
||||||
|
TARGET,
|
||||||
|
StringInput::from(source_file.as_ref()),
|
||||||
|
Some(&comments),
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut tokens: Vec<LexedItem> = lexer
|
||||||
|
.map(|token| LexedItem {
|
||||||
|
span: token.span,
|
||||||
|
inner: TokenOrComment::Token(token.token),
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
tokens.extend(flatten_comments(comments).map(|comment| LexedItem {
|
||||||
|
span: comment.span,
|
||||||
|
inner: TokenOrComment::Comment {
|
||||||
|
kind: comment.kind,
|
||||||
|
text: comment.text,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
tokens.sort_by_key(|item| item.span.lo.0);
|
||||||
|
|
||||||
|
tokens
|
||||||
|
}
|
||||||
|
|
||||||
/// A low level function which transpiles a source module into an swc
|
/// A low level function which transpiles a source module into an swc
|
||||||
/// SourceFile.
|
/// SourceFile.
|
||||||
pub fn transpile_module(
|
pub fn transpile_module(
|
||||||
|
|
|
@ -1,14 +1,15 @@
|
||||||
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
// Copyright 2018-2020 the Deno authors. All rights reserved. MIT license.
|
||||||
|
|
||||||
|
use crate::ast;
|
||||||
|
use crate::ast::TokenOrComment;
|
||||||
use crate::colors;
|
use crate::colors;
|
||||||
use crate::inspector::InspectorSession;
|
use crate::inspector::InspectorSession;
|
||||||
|
use crate::media_type::MediaType;
|
||||||
use crate::program_state::ProgramState;
|
use crate::program_state::ProgramState;
|
||||||
use crate::worker::MainWorker;
|
use crate::worker::MainWorker;
|
||||||
use deno_core::error::AnyError;
|
use deno_core::error::AnyError;
|
||||||
use deno_core::serde_json::json;
|
use deno_core::serde_json::json;
|
||||||
use deno_core::serde_json::Value;
|
use deno_core::serde_json::Value;
|
||||||
use regex::Captures;
|
|
||||||
use regex::Regex;
|
|
||||||
use rustyline::completion::Completer;
|
use rustyline::completion::Completer;
|
||||||
use rustyline::error::ReadlineError;
|
use rustyline::error::ReadlineError;
|
||||||
use rustyline::highlight::Highlighter;
|
use rustyline::highlight::Highlighter;
|
||||||
|
@ -26,6 +27,7 @@ use std::sync::mpsc::Sender;
|
||||||
use std::sync::mpsc::SyncSender;
|
use std::sync::mpsc::SyncSender;
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use std::sync::Mutex;
|
use std::sync::Mutex;
|
||||||
|
use swc_ecmascript::parser::token::{Token, Word};
|
||||||
|
|
||||||
// Provides helpers to the editor like validation for multi-line edits, completion candidates for
|
// Provides helpers to the editor like validation for multi-line edits, completion candidates for
|
||||||
// tab completion.
|
// tab completion.
|
||||||
|
@ -132,38 +134,23 @@ impl Validator for Helper {
|
||||||
&self,
|
&self,
|
||||||
ctx: &mut ValidationContext,
|
ctx: &mut ValidationContext,
|
||||||
) -> Result<ValidationResult, ReadlineError> {
|
) -> Result<ValidationResult, ReadlineError> {
|
||||||
let mut stack: Vec<char> = Vec::new();
|
let mut stack: Vec<Token> = Vec::new();
|
||||||
let mut literal: Option<char> = None;
|
let mut in_template = false;
|
||||||
let mut escape: bool = false;
|
|
||||||
|
|
||||||
for c in ctx.input().chars() {
|
for item in ast::lex("", ctx.input(), &MediaType::JavaScript) {
|
||||||
if escape {
|
if let TokenOrComment::Token(token) = item.inner {
|
||||||
escape = false;
|
match token {
|
||||||
continue;
|
Token::BackQuote => in_template = !in_template,
|
||||||
}
|
Token::LParen
|
||||||
|
| Token::LBracket
|
||||||
if c == '\\' {
|
| Token::LBrace
|
||||||
escape = true;
|
| Token::DollarLBrace => stack.push(token),
|
||||||
continue;
|
Token::RParen | Token::RBracket | Token::RBrace => {
|
||||||
}
|
match (stack.pop(), token) {
|
||||||
|
(Some(Token::LParen), Token::RParen)
|
||||||
if let Some(v) = literal {
|
| (Some(Token::LBracket), Token::RBracket)
|
||||||
if c == v {
|
| (Some(Token::LBrace), Token::RBrace)
|
||||||
literal = None
|
| (Some(Token::DollarLBrace), Token::RBrace) => {}
|
||||||
}
|
|
||||||
|
|
||||||
continue;
|
|
||||||
} else {
|
|
||||||
literal = match c {
|
|
||||||
'`' | '"' | '/' | '\'' => Some(c),
|
|
||||||
_ => None,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
match c {
|
|
||||||
'(' | '[' | '{' => stack.push(c),
|
|
||||||
')' | ']' | '}' => match (stack.pop(), c) {
|
|
||||||
(Some('('), ')') | (Some('['), ']') | (Some('{'), '}') => {}
|
|
||||||
(Some(left), _) => {
|
(Some(left), _) => {
|
||||||
return Ok(ValidationResult::Invalid(Some(format!(
|
return Ok(ValidationResult::Invalid(Some(format!(
|
||||||
"Mismatched pairs: {:?} is not properly closed",
|
"Mismatched pairs: {:?} is not properly closed",
|
||||||
|
@ -175,12 +162,14 @@ impl Validator for Helper {
|
||||||
// Thus marked as valid with no info.
|
// Thus marked as valid with no info.
|
||||||
return Ok(ValidationResult::Valid(None));
|
return Ok(ValidationResult::Valid(None));
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if !stack.is_empty() || literal == Some('`') {
|
if !stack.is_empty() || in_template {
|
||||||
return Ok(ValidationResult::Incomplete);
|
return Ok(ValidationResult::Incomplete);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -210,71 +199,60 @@ impl Highlighter for Helper {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct LineHighlighter {
|
struct LineHighlighter;
|
||||||
regex: Regex,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl LineHighlighter {
|
impl LineHighlighter {
|
||||||
fn new() -> Self {
|
fn new() -> Self {
|
||||||
let regex = Regex::new(
|
Self
|
||||||
r#"(?x)
|
|
||||||
(?P<comment>(?:/\*[\s\S]*?\*/|//[^\n]*)) |
|
|
||||||
(?P<string>(?:"([^"\\]|\\.)*"|'([^'\\]|\\.)*'|`([^`\\]|\\.)*`)) |
|
|
||||||
(?P<regexp>/(?:(?:\\/|[^\n/]))*?/[gimsuy]*) |
|
|
||||||
(?P<number>\b\d+(?:\.\d+)?(?:e[+-]?\d+)*n?\b) |
|
|
||||||
(?P<infinity>\b(?:Infinity|NaN)\b) |
|
|
||||||
(?P<hexnumber>\b0x[a-fA-F0-9]+\b) |
|
|
||||||
(?P<octalnumber>\b0o[0-7]+\b) |
|
|
||||||
(?P<binarynumber>\b0b[01]+\b) |
|
|
||||||
(?P<boolean>\b(?:true|false)\b) |
|
|
||||||
(?P<null>\b(?:null)\b) |
|
|
||||||
(?P<undefined>\b(?:undefined)\b) |
|
|
||||||
(?P<keyword>\b(?:await|async|var|let|for|if|else|in|of|class|const|function|yield|return|with|case|break|switch|import|export|new|while|do|throw|catch|this)\b) |
|
|
||||||
"#,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
Self { regex }
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Highlighter for LineHighlighter {
|
impl Highlighter for LineHighlighter {
|
||||||
fn highlight<'l>(&self, line: &'l str, _: usize) -> Cow<'l, str> {
|
fn highlight<'l>(&self, line: &'l str, _: usize) -> Cow<'l, str> {
|
||||||
self
|
let mut out_line = String::from(line);
|
||||||
.regex
|
|
||||||
.replace_all(&line.to_string(), |caps: &Captures<'_>| {
|
for item in ast::lex("", line, &MediaType::JavaScript) {
|
||||||
if let Some(cap) = caps.name("comment") {
|
// Adding color adds more bytes to the string,
|
||||||
colors::gray(cap.as_str()).to_string()
|
// so an offset is needed to stop spans falling out of sync.
|
||||||
} else if let Some(cap) = caps.name("string") {
|
let offset = out_line.len() - line.len();
|
||||||
colors::green(cap.as_str()).to_string()
|
let span = item.span_as_range();
|
||||||
} else if let Some(cap) = caps.name("regexp") {
|
|
||||||
colors::red(cap.as_str()).to_string()
|
out_line.replace_range(
|
||||||
} else if let Some(cap) = caps.name("number") {
|
span.start + offset..span.end + offset,
|
||||||
colors::yellow(cap.as_str()).to_string()
|
&match item.inner {
|
||||||
} else if let Some(cap) = caps.name("boolean") {
|
TokenOrComment::Token(token) => match token {
|
||||||
colors::yellow(cap.as_str()).to_string()
|
Token::Str { .. } | Token::Template { .. } | Token::BackQuote => {
|
||||||
} else if let Some(cap) = caps.name("null") {
|
colors::green(&line[span]).to_string()
|
||||||
colors::yellow(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("undefined") {
|
|
||||||
colors::gray(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("keyword") {
|
|
||||||
colors::cyan(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("infinity") {
|
|
||||||
colors::yellow(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("classes") {
|
|
||||||
colors::green_bold(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("hexnumber") {
|
|
||||||
colors::yellow(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("octalnumber") {
|
|
||||||
colors::yellow(cap.as_str()).to_string()
|
|
||||||
} else if let Some(cap) = caps.name("binarynumber") {
|
|
||||||
colors::yellow(cap.as_str()).to_string()
|
|
||||||
} else {
|
|
||||||
caps[0].to_string()
|
|
||||||
}
|
}
|
||||||
})
|
Token::Regex(_, _) => colors::red(&line[span]).to_string(),
|
||||||
.to_string()
|
Token::Num(_) | Token::BigInt(_) => {
|
||||||
.into()
|
colors::yellow(&line[span]).to_string()
|
||||||
|
}
|
||||||
|
Token::Word(word) => match word {
|
||||||
|
Word::True | Word::False | Word::Null => {
|
||||||
|
colors::yellow(&line[span]).to_string()
|
||||||
|
}
|
||||||
|
Word::Keyword(_) => colors::cyan(&line[span]).to_string(),
|
||||||
|
Word::Ident(ident) => {
|
||||||
|
if ident == *"undefined" {
|
||||||
|
colors::gray(&line[span]).to_string()
|
||||||
|
} else if ident == *"Infinity" || ident == *"NaN" {
|
||||||
|
colors::yellow(&line[span]).to_string()
|
||||||
|
} else {
|
||||||
|
line[span].to_string()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => line[span].to_string(),
|
||||||
|
},
|
||||||
|
TokenOrComment::Comment { .. } => {
|
||||||
|
colors::gray(&line[span]).to_string()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
out_line.into()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue