mago_syntax/parser/
mod.rs

1use mago_database::file::File;
2use mago_database::file::HasFileId;
3use mago_interner::ThreadedInterner;
4use mago_syntax_core::input::Input;
5
6use crate::ast::Program;
7use crate::ast::sequence::Sequence;
8use crate::lexer::Lexer;
9
10use crate::error::ParseError;
11use crate::parser::internal::statement::parse_statement;
12use crate::parser::internal::token_stream::TokenStream;
13
14mod internal;
15
16pub fn parse_file(interner: &ThreadedInterner, file: &File) -> (Program, Option<ParseError>) {
17    let lexer = Lexer::new(interner, Input::from_file(file));
18
19    construct(interner, lexer)
20}
21
22pub fn parse(interner: &ThreadedInterner, input: Input<'_>) -> (Program, Option<ParseError>) {
23    let lexer = Lexer::new(interner, input);
24
25    construct(interner, lexer)
26}
27
28fn construct<'i>(interner: &'i ThreadedInterner, lexer: Lexer<'_, 'i>) -> (Program, Option<ParseError>) {
29    let mut stream = TokenStream::new(interner, lexer);
30
31    let mut error = None;
32    let statements = {
33        let mut statements = Vec::new();
34
35        loop {
36            match stream.has_reached_eof() {
37                Ok(false) => match parse_statement(&mut stream) {
38                    Ok(statement) => {
39                        statements.push(statement);
40                    }
41                    Err(parse_error) => {
42                        error = Some(parse_error);
43
44                        break;
45                    }
46                },
47                Ok(true) => {
48                    break;
49                }
50                Err(syntax_error) => {
51                    error = Some(ParseError::from(syntax_error));
52
53                    break;
54                }
55            }
56        }
57
58        statements
59    };
60
61    (Program { file_id: stream.file_id(), statements: Sequence::new(statements), trivia: stream.get_trivia() }, error)
62}