use super::left_hand_side::LeftHandSideExpression;
use crate::syntax::{
ast::{node, op::UnaryOp, Node, Punctuator},
lexer::{Error as LexError, TokenKind},
parser::{
expression::unary::UnaryExpression, AllowAwait, AllowYield, Cursor, ParseError,
ParseResult, TokenParser,
},
};
use boa_interner::{Interner, Sym};
use boa_profiler::Profiler;
use std::io::Read;
#[derive(Debug, Clone, Copy)]
pub(super) struct UpdateExpression {
name: Option<Sym>,
allow_yield: AllowYield,
allow_await: AllowAwait,
}
impl UpdateExpression {
pub(super) fn new<N, Y, A>(name: N, allow_yield: Y, allow_await: A) -> Self
where
N: Into<Option<Sym>>,
Y: Into<AllowYield>,
A: Into<AllowAwait>,
{
Self {
name: name.into(),
allow_yield: allow_yield.into(),
allow_await: allow_await.into(),
}
}
}
impl<R> TokenParser<R> for UpdateExpression
where
R: Read,
{
type Output = Node;
fn parse(self, cursor: &mut Cursor<R>, interner: &mut Interner) -> ParseResult {
let _timer = Profiler::global().start_event("UpdateExpression", "Parsing");
let tok = cursor.peek(0, interner)?.ok_or(ParseError::AbruptEnd)?;
let position = tok.span().start();
match tok.kind() {
TokenKind::Punctuator(Punctuator::Inc) => {
cursor
.next(interner)?
.expect("Punctuator::Inc token disappeared");
let target = UnaryExpression::new(self.name, self.allow_yield, self.allow_await)
.parse(cursor, interner)?;
if cursor.strict_mode() {
if let Node::Identifier(ident) = target {
ident.check_strict_arguments_or_eval(position)?;
}
}
return Ok(node::UnaryOp::new(UnaryOp::IncrementPre, target).into());
}
TokenKind::Punctuator(Punctuator::Dec) => {
cursor
.next(interner)?
.expect("Punctuator::Dec token disappeared");
let target = UnaryExpression::new(self.name, self.allow_yield, self.allow_await)
.parse(cursor, interner)?;
if cursor.strict_mode() {
if let Node::Identifier(ident) = target {
ident.check_strict_arguments_or_eval(position)?;
}
}
return Ok(node::UnaryOp::new(UnaryOp::DecrementPre, target).into());
}
_ => {}
}
let lhs = LeftHandSideExpression::new(self.name, self.allow_yield, self.allow_await)
.parse(cursor, interner)?;
let strict = cursor.strict_mode();
if let Some(tok) = cursor.peek(0, interner)? {
let token_start = tok.span().start();
match tok.kind() {
TokenKind::Punctuator(Punctuator::Inc) => {
cursor
.next(interner)?
.expect("Punctuator::Inc token disappeared");
let ok = match &lhs {
Node::Identifier(_) if !strict => true,
Node::Identifier(ident)
if ![Sym::EVAL, Sym::ARGUMENTS].contains(&ident.sym()) =>
{
true
}
Node::GetConstField(_) | Node::GetField(_) => true,
_ => false,
};
if !ok {
return Err(ParseError::lex(LexError::Syntax(
"Invalid left-hand side in assignment".into(),
token_start,
)));
}
return Ok(node::UnaryOp::new(UnaryOp::IncrementPost, lhs).into());
}
TokenKind::Punctuator(Punctuator::Dec) => {
cursor
.next(interner)?
.expect("Punctuator::Dec token disappeared");
let ok = match &lhs {
Node::Identifier(_) if !strict => true,
Node::Identifier(ident)
if ![Sym::EVAL, Sym::ARGUMENTS].contains(&ident.sym()) =>
{
true
}
Node::GetConstField(_) | Node::GetField(_) => true,
_ => false,
};
if !ok {
return Err(ParseError::lex(LexError::Syntax(
"Invalid left-hand side in assignment".into(),
token_start,
)));
}
return Ok(node::UnaryOp::new(UnaryOp::DecrementPost, lhs).into());
}
_ => {}
}
}
Ok(lhs)
}
}