From be036958687f24faddabfd838490a868e47b6b0e Mon Sep 17 00:00:00 2001 From: croraf Date: Thu, 4 Jun 2020 17:29:34 +0200 Subject: [PATCH 1/2] Basic template literal lexer implementation --- boa/src/syntax/ast/token.rs | 11 +++++++++++ boa/src/syntax/lexer/mod.rs | 23 +++++++++++++++++++++++ boa/src/syntax/lexer/tests.rs | 20 ++++++++++++++++++++ 3 files changed, 54 insertions(+) diff --git a/boa/src/syntax/ast/token.rs b/boa/src/syntax/ast/token.rs index dafcc0ae92e..eff2688f971 100644 --- a/boa/src/syntax/ast/token.rs +++ b/boa/src/syntax/ast/token.rs @@ -239,6 +239,8 @@ pub enum TokenKind { /// A string literal. StringLiteral(Box), + TemplateLiteral(Box), + /// A regular expression, consisting of body and flags. RegularExpressionLiteral(Box, RegExpFlags), @@ -309,6 +311,14 @@ impl TokenKind { Self::StringLiteral(lit.into()) } + /// Creates a `TemplateLiteral` token type. + pub fn template_literal(lit: S) -> Self + where + S: Into>, + { + Self::TemplateLiteral(lit.into()) + } + /// Creates a `RegularExpressionLiteral` token kind. pub fn regular_expression_literal(body: B, flags: RegExpFlags) -> Self where @@ -336,6 +346,7 @@ impl Display for TokenKind { Self::NumericLiteral(NumericLiteral::BigInt(ref num)) => write!(f, "{}n", num), Self::Punctuator(ref punc) => write!(f, "{}", punc), Self::StringLiteral(ref lit) => write!(f, "{}", lit), + Self::TemplateLiteral(ref lit) => write!(f, "{}", lit), Self::RegularExpressionLiteral(ref body, ref flags) => write!(f, "/{}/{}", body, flags), Self::LineTerminator => write!(f, "line terminator"), } diff --git a/boa/src/syntax/lexer/mod.rs b/boa/src/syntax/lexer/mod.rs index 25431ed4819..78bfc10a92d 100644 --- a/boa/src/syntax/lexer/mod.rs +++ b/boa/src/syntax/lexer/mod.rs @@ -495,6 +495,7 @@ impl<'a> Lexer<'a> { self.next_column(); let ch = self.next(); match ch { + // StringLiteral '"' | '\'' => { let mut buf = String::new(); loop { @@ -619,6 +620,28 @@ impl<'a> Lexer<'a> { self.move_columns( str_length.wrapping_add(1)); self.push_token(TokenKind::string_literal(buf), start_pos); } + // TemplateLiteral + '`' => { + let mut buf = String::new(); + loop { + if self.preview_next().is_none() { + return Err(LexerError::new("Unterminated template literal")); + } + match self.next() { + '`' => { + break; + } + next_ch => buf.push(next_ch), + // TODO when there is an expression inside the literal + } + } + let str_length = buf.len() as u32; + // Why +1? Quotation marks are not included, + // So technically it would be +2, (for both " ") but we want to be 1 less + // to compensate for the incrementing at the top + self.move_columns( str_length.wrapping_add(1)); + self.push_token(TokenKind::template_literal(buf), start_pos); + } _ if ch.is_digit(10) => self.reed_numerical_literal(ch)?, _ if ch.is_alphabetic() || ch == '$' || ch == '_' => { let mut buf = ch.to_string(); diff --git a/boa/src/syntax/lexer/tests.rs b/boa/src/syntax/lexer/tests.rs index 6f8c1e1086f..2143d7af00c 100644 --- a/boa/src/syntax/lexer/tests.rs +++ b/boa/src/syntax/lexer/tests.rs @@ -37,6 +37,26 @@ fn check_string() { assert_eq!(lexer.tokens[1].kind, TokenKind::string_literal("bbb")); } +#[test] +fn check_template_literal_simple() { + let s = "`I'm a template literal`"; + let mut lexer = Lexer::new(s); + lexer.lex().expect("failed to lex"); + assert_eq!(lexer.tokens[0].kind, TokenKind::template_literal("I'm a template literal")); +} + +#[test] +fn check_template_literal_unterminated() { + let s = "`I'm a template"; + let mut lexer = Lexer::new(s); + match lexer.lex() { + Ok(_) => panic!("Lexer did not detect end of stream"), + Err(e) => { + assert_eq!(e.to_string(), "Unterminated template literal"); + } + } +} + #[test] fn check_punctuators() { // https://tc39.es/ecma262/#sec-punctuators From f3ec4e2c9460c16fd804c7e80d3d097a95fcd72b Mon Sep 17 00:00:00 2001 From: croraf Date: Thu, 4 Jun 2020 17:48:47 +0200 Subject: [PATCH 2/2] Fix formatting --- boa/src/syntax/lexer/tests.rs | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/boa/src/syntax/lexer/tests.rs b/boa/src/syntax/lexer/tests.rs index 2143d7af00c..4129bb3c179 100644 --- a/boa/src/syntax/lexer/tests.rs +++ b/boa/src/syntax/lexer/tests.rs @@ -42,7 +42,10 @@ fn check_template_literal_simple() { let s = "`I'm a template literal`"; let mut lexer = Lexer::new(s); lexer.lex().expect("failed to lex"); - assert_eq!(lexer.tokens[0].kind, TokenKind::template_literal("I'm a template literal")); + assert_eq!( + lexer.tokens[0].kind, + TokenKind::template_literal("I'm a template literal") + ); } #[test]