mago-syntax 1.0.0-alpha.5

A correct, fast, and memory-efficient PHP syntax implementation, including Lexer, Parser, AST, and utilities for Mago.
Documentation
use crate::T;
use crate::ast::ast::*;
use crate::ast::sequence::TokenSeparatedSequence;
use crate::error::ParseError;
use crate::parser::internal::expression::parse_expression;
use crate::parser::internal::token_stream::TokenStream;
use crate::parser::internal::utils;

pub fn parse_match(stream: &mut TokenStream<'_, '_>) -> Result<Match, ParseError> {
    Ok(Match {
        r#match: utils::expect_keyword(stream, T!["match"])?,
        left_parenthesis: utils::expect_span(stream, T!["("])?,
        expression: Box::new(parse_expression(stream)?),
        right_parenthesis: utils::expect_span(stream, T![")"])?,
        left_brace: utils::expect_span(stream, T!["{"])?,
        arms: {
            let mut arms = vec![];
            let mut commas = vec![];
            loop {
                if matches!(utils::peek(stream)?.kind, T!["}"]) {
                    break;
                }

                arms.push(parse_match_arm(stream)?);

                match utils::peek(stream)?.kind {
                    T![","] => {
                        commas.push(utils::expect_any(stream)?);
                    }
                    _ => {
                        break;
                    }
                }
            }

            TokenSeparatedSequence::new(arms, commas)
        },
        right_brace: utils::expect_span(stream, T!["}"])?,
    })
}

pub fn parse_match_arm(stream: &mut TokenStream<'_, '_>) -> Result<MatchArm, ParseError> {
    Ok(match utils::peek(stream)?.kind {
        T!["default"] => MatchArm::Default(parse_match_default_arm(stream)?),
        _ => MatchArm::Expression(parse_match_expression_arm(stream)?),
    })
}

pub fn parse_match_expression_arm(stream: &mut TokenStream<'_, '_>) -> Result<MatchExpressionArm, ParseError> {
    Ok(MatchExpressionArm {
        conditions: {
            let mut conditions = vec![];
            let mut commas = vec![];
            loop {
                if matches!(utils::peek(stream)?.kind, T!["=>"]) {
                    break;
                }

                conditions.push(parse_expression(stream)?);

                match utils::peek(stream)?.kind {
                    T![","] => {
                        commas.push(utils::expect_any(stream)?);
                    }
                    _ => {
                        break;
                    }
                }
            }

            TokenSeparatedSequence::new(conditions, commas)
        },
        arrow: utils::expect_span(stream, T!["=>"])?,
        expression: Box::new(parse_expression(stream)?),
    })
}

pub fn parse_match_default_arm(stream: &mut TokenStream<'_, '_>) -> Result<MatchDefaultArm, ParseError> {
    Ok(MatchDefaultArm {
        default: utils::expect_keyword(stream, T!["default"])?,
        arrow: utils::expect_span(stream, T!["=>"])?,
        expression: Box::new(parse_expression(stream)?),
    })
}