1use cairo_lang_diagnostics::{Diagnostics, DiagnosticsBuilder, Maybe, ToMaybe};
2use cairo_lang_filesystem::db::FilesGroup;
3use cairo_lang_filesystem::ids::{FileId, FileKind};
4use cairo_lang_syntax::node::ast::{Expr, StatementList, SyntaxFile};
5use cairo_lang_syntax::node::{SyntaxNode, TypedSyntaxNode};
6use salsa::Database;
7
8use crate::diagnostic::ParserDiagnostic;
9use crate::parser::Parser;
10
11#[cfg(test)]
12#[path = "db_test.rs"]
13mod db_test;
14
15pub trait ParserGroup: Database {
17 fn file_syntax<'db>(&'db self, file_id: FileId<'db>) -> Maybe<SyntaxNode<'db>> {
19 file_syntax(self.as_dyn_database(), file_id)
20 }
21
22 fn file_module_syntax<'db>(&'db self, file_id: FileId<'db>) -> Maybe<SyntaxFile<'db>> {
24 file_module_syntax(self.as_dyn_database(), file_id)
25 }
26 fn file_expr_syntax<'db>(&'db self, file_id: FileId<'db>) -> Maybe<Expr<'db>> {
29 file_expr_syntax(self.as_dyn_database(), file_id)
30 }
31 fn file_statement_list_syntax<'db>(
34 &'db self,
35 file_id: FileId<'db>,
36 ) -> Maybe<StatementList<'db>> {
37 file_statement_list_syntax(self.as_dyn_database(), file_id)
38 }
39 fn file_syntax_diagnostics<'db>(
41 &'db self,
42 file_id: FileId<'db>,
43 ) -> &'db Diagnostics<'db, ParserDiagnostic<'db>> {
44 file_syntax_diagnostics(self.as_dyn_database(), file_id)
45 }
46}
47
48impl<T: Database + ?Sized> ParserGroup for T {}
49
50#[salsa::tracked]
51struct SyntaxData<'db> {
52 diagnostics: Diagnostics<'db, ParserDiagnostic<'db>>,
53 syntax: Maybe<SyntaxNode<'db>>,
54}
55
56#[salsa::tracked(returns(ref))]
58fn file_syntax_data<'db>(db: &'db dyn Database, file_id: FileId<'db>) -> SyntaxData<'db> {
59 let mut diagnostics = DiagnosticsBuilder::default();
60 let syntax = db.file_content(file_id).to_maybe().map(|s| match file_id.kind(db) {
61 FileKind::Module => Parser::parse_file(db, &mut diagnostics, file_id, s).as_syntax_node(),
62 FileKind::Expr => {
63 Parser::parse_file_expr(db, &mut diagnostics, file_id, s).as_syntax_node()
64 }
65 FileKind::StatementList => {
66 Parser::parse_file_statement_list(db, &mut diagnostics, file_id, s).as_syntax_node()
67 }
68 });
69 SyntaxData::new(db, diagnostics.build(), syntax)
70}
71
72#[salsa::tracked]
74fn file_syntax<'db>(db: &'db dyn Database, file_id: FileId<'db>) -> Maybe<SyntaxNode<'db>> {
75 file_syntax_data(db, file_id).syntax(db)
76}
77
78fn file_module_syntax<'db>(db: &'db dyn Database, file_id: FileId<'db>) -> Maybe<SyntaxFile<'db>> {
79 assert_eq!(file_id.kind(db), FileKind::Module, "file_id must be a module");
80 Ok(SyntaxFile::from_syntax_node(db, file_syntax(db, file_id)?))
81}
82
83fn file_expr_syntax<'db>(db: &'db dyn Database, file_id: FileId<'db>) -> Maybe<Expr<'db>> {
84 assert_eq!(file_id.kind(db), FileKind::Expr, "file_id must be an expr");
85 Ok(Expr::from_syntax_node(db, file_syntax(db, file_id)?))
86}
87
88fn file_statement_list_syntax<'db>(
89 db: &'db dyn Database,
90 file_id: FileId<'db>,
91) -> Maybe<StatementList<'db>> {
92 assert_eq!(file_id.kind(db), FileKind::StatementList, "file_id must be a statement list");
93 Ok(StatementList::from_syntax_node(db, file_syntax(db, file_id)?))
94}
95
96#[salsa::tracked(returns(ref))]
97fn file_syntax_diagnostics<'db>(
98 db: &'db dyn Database,
99 file_id: FileId<'db>,
100) -> Diagnostics<'db, ParserDiagnostic<'db>> {
101 file_syntax_data(db, file_id).diagnostics(db).clone()
102}