mirror of
https://github.com/danbulant/oxc
synced 2026-05-21 13:18:59 +00:00
289 lines
9.6 KiB
Rust
289 lines
9.6 KiB
Rust
use std::cell::Cell;
|
|
|
|
use oxc_allocator::Box;
|
|
use oxc_ast::ast::*;
|
|
use oxc_diagnostics::Result;
|
|
use oxc_span::Span;
|
|
|
|
use super::list::FormalParameterList;
|
|
use crate::{diagnostics, lexer::Kind, list::SeparatedList, Context, ParserImpl, StatementContext};
|
|
|
|
#[derive(Debug, Clone, Copy, Eq, PartialEq)]
|
|
pub enum FunctionKind {
|
|
Declaration { single_statement: bool },
|
|
Expression,
|
|
DefaultExport,
|
|
TSDeclaration,
|
|
}
|
|
|
|
impl FunctionKind {
|
|
pub(crate) fn is_id_required(self) -> bool {
|
|
matches!(self, Self::Declaration { single_statement: true })
|
|
}
|
|
|
|
pub(crate) fn is_expression(self) -> bool {
|
|
self == Self::Expression
|
|
}
|
|
}
|
|
|
|
impl<'a> ParserImpl<'a> {
|
|
pub(crate) fn at_function_with_async(&mut self) -> bool {
|
|
self.at(Kind::Function)
|
|
|| self.at(Kind::Async)
|
|
&& self.peek_at(Kind::Function)
|
|
&& !self.peek_token().is_on_new_line
|
|
}
|
|
|
|
pub(crate) fn at_async_no_new_line(&mut self) -> bool {
|
|
self.at(Kind::Async) && !self.cur_token().escaped() && !self.peek_token().is_on_new_line
|
|
}
|
|
|
|
pub(crate) fn parse_function_body(&mut self) -> Result<Box<'a, FunctionBody<'a>>> {
|
|
let span = self.start_span();
|
|
self.expect(Kind::LCurly)?;
|
|
|
|
let (directives, statements) = self.context(Context::Return, Context::empty(), |p| {
|
|
p.parse_directives_and_statements(/* is_top_level */ false)
|
|
})?;
|
|
|
|
self.expect(Kind::RCurly)?;
|
|
Ok(self.ast.function_body(self.end_span(span), directives, statements))
|
|
}
|
|
|
|
pub(crate) fn parse_formal_parameters(
|
|
&mut self,
|
|
params_kind: FormalParameterKind,
|
|
) -> Result<(Option<TSThisParameter<'a>>, Box<'a, FormalParameters<'a>>)> {
|
|
let span = self.start_span();
|
|
let list: FormalParameterList<'_> = FormalParameterList::parse(self)?;
|
|
let formal_parameters =
|
|
self.ast.formal_parameters(self.end_span(span), params_kind, list.elements, list.rest);
|
|
let this_param = list.this_param;
|
|
Ok((this_param, formal_parameters))
|
|
}
|
|
|
|
pub(crate) fn parse_function(
|
|
&mut self,
|
|
span: Span,
|
|
id: Option<BindingIdentifier<'a>>,
|
|
r#async: bool,
|
|
generator: bool,
|
|
func_kind: FunctionKind,
|
|
modifiers: Modifiers<'a>,
|
|
) -> Result<Box<'a, Function<'a>>> {
|
|
let ctx = self.ctx;
|
|
self.ctx = self.ctx.and_in(true).and_await(r#async).and_yield(generator);
|
|
|
|
let type_parameters = self.parse_ts_type_parameters()?;
|
|
|
|
let (this_param, params) =
|
|
self.parse_formal_parameters(FormalParameterKind::FormalParameter)?;
|
|
|
|
let return_type = self.parse_ts_return_type_annotation()?;
|
|
|
|
let body = if self.at(Kind::LCurly) { Some(self.parse_function_body()?) } else { None };
|
|
|
|
self.ctx =
|
|
self.ctx.and_in(ctx.has_in()).and_await(ctx.has_await()).and_yield(ctx.has_yield());
|
|
|
|
if !self.ts_enabled() && body.is_none() {
|
|
return Err(self.unexpected());
|
|
}
|
|
|
|
let function_type = match func_kind {
|
|
FunctionKind::Declaration { .. } | FunctionKind::DefaultExport => {
|
|
if body.is_none() {
|
|
FunctionType::TSDeclareFunction
|
|
} else {
|
|
FunctionType::FunctionDeclaration
|
|
}
|
|
}
|
|
FunctionKind::Expression { .. } => {
|
|
if body.is_none() {
|
|
FunctionType::TSEmptyBodyFunctionExpression
|
|
} else {
|
|
FunctionType::FunctionExpression
|
|
}
|
|
}
|
|
FunctionKind::TSDeclaration { .. } => FunctionType::TSDeclareFunction,
|
|
};
|
|
|
|
if FunctionType::TSDeclareFunction == function_type
|
|
|| FunctionType::TSEmptyBodyFunctionExpression == function_type
|
|
{
|
|
self.asi()?;
|
|
}
|
|
|
|
Ok(self.ast.function(
|
|
function_type,
|
|
self.end_span(span),
|
|
id,
|
|
generator,
|
|
r#async,
|
|
this_param,
|
|
params,
|
|
body,
|
|
type_parameters,
|
|
return_type,
|
|
modifiers,
|
|
))
|
|
}
|
|
|
|
/// [Function Declaration](https://tc39.es/ecma262/#prod-FunctionDeclaration)
|
|
pub(crate) fn parse_function_declaration(
|
|
&mut self,
|
|
stmt_ctx: StatementContext,
|
|
) -> Result<Statement<'a>> {
|
|
let func_kind =
|
|
FunctionKind::Declaration { single_statement: stmt_ctx.is_single_statement() };
|
|
let decl = self.parse_function_impl(func_kind)?;
|
|
if stmt_ctx.is_single_statement() {
|
|
if decl.r#async {
|
|
self.error(diagnostics::async_function_declaration(Span::new(
|
|
decl.span.start,
|
|
decl.params.span.end,
|
|
)));
|
|
} else if decl.generator {
|
|
self.error(diagnostics::generator_function_declaration(Span::new(
|
|
decl.span.start,
|
|
decl.params.span.end,
|
|
)));
|
|
}
|
|
}
|
|
|
|
Ok(self.ast.function_declaration(decl))
|
|
}
|
|
|
|
/// Parse function implementation in Javascript, cursor
|
|
/// at `function` or `async function`
|
|
pub(crate) fn parse_function_impl(
|
|
&mut self,
|
|
func_kind: FunctionKind,
|
|
) -> Result<Box<'a, Function<'a>>> {
|
|
let span = self.start_span();
|
|
let r#async = self.eat(Kind::Async);
|
|
self.expect(Kind::Function)?;
|
|
let generator = self.eat(Kind::Star);
|
|
let id = self.parse_function_id(func_kind, r#async, generator);
|
|
self.parse_function(span, id, r#async, generator, func_kind, Modifiers::empty())
|
|
}
|
|
|
|
/// Parse function implementation in Typescript, cursor
|
|
/// at `function`
|
|
pub(crate) fn parse_ts_function_impl(
|
|
&mut self,
|
|
start_span: Span,
|
|
func_kind: FunctionKind,
|
|
modifiers: Modifiers<'a>,
|
|
) -> Result<Box<'a, Function<'a>>> {
|
|
let r#async = modifiers.contains(ModifierKind::Async);
|
|
self.expect(Kind::Function)?;
|
|
let generator = self.eat(Kind::Star);
|
|
let id = self.parse_function_id(func_kind, r#async, generator);
|
|
self.parse_function(start_span, id, r#async, generator, func_kind, modifiers)
|
|
}
|
|
|
|
/// [Function Expression](https://tc39.es/ecma262/#prod-FunctionExpression)
|
|
pub(crate) fn parse_function_expression(
|
|
&mut self,
|
|
span: Span,
|
|
r#async: bool,
|
|
) -> Result<Expression<'a>> {
|
|
let func_kind = FunctionKind::Expression;
|
|
self.expect(Kind::Function)?;
|
|
|
|
let generator = self.eat(Kind::Star);
|
|
let id = self.parse_function_id(func_kind, r#async, generator);
|
|
let function =
|
|
self.parse_function(span, id, r#async, generator, func_kind, Modifiers::empty())?;
|
|
|
|
Ok(self.ast.function_expression(function))
|
|
}
|
|
|
|
/// Section 15.4 Method Definitions
|
|
/// `ClassElementName` ( `UniqueFormalParameters` ) { `FunctionBody` }
|
|
/// `GeneratorMethod`
|
|
/// * `ClassElementName`
|
|
/// `AsyncMethod`
|
|
/// async `ClassElementName`
|
|
/// `AsyncGeneratorMethod`
|
|
/// async * `ClassElementName`
|
|
pub(crate) fn parse_method(
|
|
&mut self,
|
|
r#async: bool,
|
|
generator: bool,
|
|
) -> Result<Box<'a, Function<'a>>> {
|
|
let span = self.start_span();
|
|
self.parse_function(
|
|
span,
|
|
None,
|
|
r#async,
|
|
generator,
|
|
FunctionKind::Expression,
|
|
Modifiers::empty(),
|
|
)
|
|
}
|
|
|
|
/// Section 15.5 Yield Expression
|
|
/// yield
|
|
/// yield [no `LineTerminator` here] `AssignmentExpression`
|
|
/// yield [no `LineTerminator` here] * `AssignmentExpression`
|
|
pub(crate) fn parse_yield_expression(&mut self) -> Result<Expression<'a>> {
|
|
let span = self.start_span();
|
|
self.bump_any(); // advance `yield`
|
|
|
|
let has_yield = self.ctx.has_yield();
|
|
if !has_yield {
|
|
self.error(diagnostics::yield_expression(Span::new(span.start, span.start + 5)));
|
|
}
|
|
|
|
let mut delegate = false;
|
|
let mut argument = None;
|
|
|
|
if !self.cur_token().is_on_new_line {
|
|
delegate = self.eat(Kind::Star);
|
|
let not_assignment_expr = matches!(
|
|
self.cur_kind(),
|
|
Kind::Semicolon
|
|
| Kind::Eof
|
|
| Kind::RCurly
|
|
| Kind::RParen
|
|
| Kind::RBrack
|
|
| Kind::Colon
|
|
| Kind::Comma
|
|
);
|
|
if !not_assignment_expr || delegate {
|
|
self.ctx = self.ctx.union_yield_if(true);
|
|
argument = Some(self.parse_assignment_expression_or_higher()?);
|
|
self.ctx = self.ctx.and_yield(has_yield);
|
|
}
|
|
}
|
|
|
|
Ok(self.ast.yield_expression(self.end_span(span), delegate, argument))
|
|
}
|
|
|
|
// id: None - for AnonymousDefaultExportedFunctionDeclaration
|
|
pub(crate) fn parse_function_id(
|
|
&mut self,
|
|
kind: FunctionKind,
|
|
r#async: bool,
|
|
generator: bool,
|
|
) -> Option<BindingIdentifier<'a>> {
|
|
let ctx = self.ctx;
|
|
if kind.is_expression() {
|
|
self.ctx = self.ctx.and_await(r#async).and_yield(generator);
|
|
}
|
|
let id = self.cur_kind().is_binding_identifier().then(|| {
|
|
let (span, name) = self.parse_identifier_kind(Kind::Ident);
|
|
self.check_identifier(span, &name);
|
|
BindingIdentifier { span, name, symbol_id: Cell::default() }
|
|
});
|
|
self.ctx = ctx;
|
|
|
|
if kind.is_id_required() && id.is_none() {
|
|
self.error(diagnostics::expect_function_name(self.cur_token().span()));
|
|
}
|
|
|
|
id
|
|
}
|
|
}
|