use super::AssignmentExpression;
use crate::syntax::{
ast::{
node::{
declaration::Declaration, ArrowFunctionDecl, FormalParameter, FormalParameterList,
FormalParameterListFlags, Node, Return, StatementList,
},
Punctuator,
},
lexer::{Error as LexError, TokenKind},
parser::{
error::{ErrorContext, ParseError, ParseResult},
expression::BindingIdentifier,
function::{FormalParameters, FunctionBody},
AllowAwait, AllowIn, AllowYield, Cursor, TokenParser,
},
};
use boa_interner::{Interner, Sym};
use boa_profiler::Profiler;
use std::io::Read;
#[derive(Debug, Clone, Copy)]
pub(in crate::syntax::parser) struct ArrowFunction {
name: Option<Sym>,
allow_in: AllowIn,
allow_yield: AllowYield,
allow_await: AllowAwait,
}
impl ArrowFunction {
pub(in crate::syntax::parser) fn new<N, I, Y, A>(
name: N,
allow_in: I,
allow_yield: Y,
allow_await: A,
) -> Self
where
N: Into<Option<Sym>>,
I: Into<AllowIn>,
Y: Into<AllowYield>,
A: Into<AllowAwait>,
{
Self {
name: name.into(),
allow_in: allow_in.into(),
allow_yield: allow_yield.into(),
allow_await: allow_await.into(),
}
}
}
impl<R> TokenParser<R> for ArrowFunction
where
R: Read,
{
type Output = ArrowFunctionDecl;
fn parse(
self,
cursor: &mut Cursor<R>,
interner: &mut Interner,
) -> Result<Self::Output, ParseError> {
let _timer = Profiler::global().start_event("ArrowFunction", "Parsing");
let next_token = cursor.peek(0, interner)?.ok_or(ParseError::AbruptEnd)?;
let (params, params_start_position) =
if let TokenKind::Punctuator(Punctuator::OpenParen) = &next_token.kind() {
let params_start_position = cursor
.expect(Punctuator::OpenParen, "arrow function", interner)?
.span()
.end();
let params = FormalParameters::new(self.allow_yield, self.allow_await)
.parse(cursor, interner)?;
cursor.expect(Punctuator::CloseParen, "arrow function", interner)?;
(params, params_start_position)
} else {
let params_start_position = next_token.span().start();
let param = BindingIdentifier::new(self.allow_yield, self.allow_await)
.parse(cursor, interner)
.context("arrow function")?;
let has_arguments = param == Sym::ARGUMENTS;
let mut flags = FormalParameterListFlags::IS_SIMPLE;
if has_arguments {
flags |= FormalParameterListFlags::HAS_ARGUMENTS;
}
(
FormalParameterList::new(
Box::new([FormalParameter::new(
Declaration::new_with_identifier(param, None),
false,
)]),
flags,
1,
),
params_start_position,
)
};
cursor.peek_expect_no_lineterminator(0, "arrow function", interner)?;
cursor.expect(
TokenKind::Punctuator(Punctuator::Arrow),
"arrow function",
interner,
)?;
let arrow = cursor.arrow();
cursor.set_arrow(true);
let body = ConciseBody::new(self.allow_in).parse(cursor, interner)?;
cursor.set_arrow(arrow);
if params.has_duplicates() {
return Err(ParseError::lex(LexError::Syntax(
"Duplicate parameter name not allowed in this context".into(),
params_start_position,
)));
}
if body.strict() && !params.is_simple() {
return Err(ParseError::lex(LexError::Syntax(
"Illegal 'use strict' directive in function with non-simple parameter list".into(),
params_start_position,
)));
}
params.name_in_lexically_declared_names(
&body.lexically_declared_names_top_level(),
params_start_position,
)?;
Ok(ArrowFunctionDecl::new(self.name, params, body))
}
}
#[derive(Debug, Clone, Copy)]
pub(in crate::syntax::parser) struct ConciseBody {
allow_in: AllowIn,
}
impl ConciseBody {
pub(in crate::syntax::parser) fn new<I>(allow_in: I) -> Self
where
I: Into<AllowIn>,
{
Self {
allow_in: allow_in.into(),
}
}
}
impl<R> TokenParser<R> for ConciseBody
where
R: Read,
{
type Output = StatementList;
fn parse(
self,
cursor: &mut Cursor<R>,
interner: &mut Interner,
) -> Result<Self::Output, ParseError> {
match cursor
.peek(0, interner)?
.ok_or(ParseError::AbruptEnd)?
.kind()
{
TokenKind::Punctuator(Punctuator::OpenBlock) => {
let _next = cursor.next(interner)?;
let body = FunctionBody::new(false, false).parse(cursor, interner)?;
cursor.expect(Punctuator::CloseBlock, "arrow function", interner)?;
Ok(body)
}
_ => Ok(StatementList::from(vec![Return::new(
ExpressionBody::new(self.allow_in, false).parse(cursor, interner)?,
None,
)
.into()])),
}
}
}
#[derive(Debug, Clone, Copy)]
struct ExpressionBody {
allow_in: AllowIn,
allow_await: AllowAwait,
}
impl ExpressionBody {
fn new<I, A>(allow_in: I, allow_await: A) -> Self
where
I: Into<AllowIn>,
A: Into<AllowAwait>,
{
Self {
allow_in: allow_in.into(),
allow_await: allow_await.into(),
}
}
}
impl<R> TokenParser<R> for ExpressionBody
where
R: Read,
{
type Output = Node;
fn parse(self, cursor: &mut Cursor<R>, interner: &mut Interner) -> ParseResult {
AssignmentExpression::new(None, self.allow_in, false, self.allow_await)
.parse(cursor, interner)
}
}