use std::path::PathBuf;
use cairo_lang_filesystem::ids::{FileId, SmolStrId};
use cairo_lang_filesystem::span::TextSpan;
use cairo_lang_syntax::node::ast::{
ModuleItemList, SyntaxFile, TerminalEndOfFile, TokenEndOfFile, Trivia,
};
use cairo_lang_syntax::node::{SyntaxNode, Terminal, Token as SyntaxToken, TypedSyntaxNode};
use indoc::indoc;
use pretty_assertions::assert_eq;
use salsa::Database;
use crate::db::ParserGroup;
use crate::printer::print_tree;
use crate::test_utils::{MockToken, MockTokenStream, create_virtual_file};
use crate::utils::{SimpleParserDatabase, get_syntax_root_and_diagnostics_from_file};
fn build_empty_file_green_tree<'a>(db: &'a dyn Database, file_id: FileId<'a>) -> SyntaxFile<'a> {
let eof_token = TokenEndOfFile::new_green(db, SmolStrId::from(db, ""));
let eof_terminal = TerminalEndOfFile::new_green(
db,
Trivia::new_green(db, &[]),
eof_token,
Trivia::new_green(db, &[]),
);
SyntaxFile::from_syntax_node(
db,
SyntaxNode::new_root(
db,
file_id,
SyntaxFile::new_green(db, ModuleItemList::new_green(db, &[]), eof_terminal).0,
),
)
}
#[test]
fn test_parser() {
let db = SimpleParserDatabase::default();
let file_id = create_virtual_file(&db, "file.cairo", "");
let syntax_file = db.file_module_syntax(file_id).unwrap();
let diagnostics = db.file_syntax_diagnostics(file_id);
assert_eq!(diagnostics.format(&db), "");
let expected_syntax_file = build_empty_file_green_tree(&db, file_id);
assert_eq!(syntax_file, expected_syntax_file);
}
#[test]
fn test_parser_shorthand() {
let db = SimpleParserDatabase::default();
let (_node, diagnostics) = db.parse_virtual_with_diagnostics("");
assert_eq!(diagnostics.format(&db), "");
let _node = db.parse_virtual("").unwrap();
}
#[test]
fn test_token_stream_parser() {
let filepath: PathBuf =
[env!("CARGO_MANIFEST_DIR"), "src/parser_test_data/cairo_test_files/short.cairo"]
.into_iter()
.collect();
let db_val = SimpleParserDatabase::default();
let db = &db_val;
let (root_node, _) = get_syntax_root_and_diagnostics_from_file(db, filepath);
let token_stream = MockTokenStream::from_syntax_node(db, root_node);
let (node_from_token_stream, _) = db.parse_token_stream(&token_stream);
let original_leaves: Vec<SyntaxNode<'_>> = root_node.tokens(db).collect();
let token_stream_origin_leaves: Vec<SyntaxNode<'_>> =
node_from_token_stream.tokens(db).collect();
assert_eq!(original_leaves.len(), token_stream_origin_leaves.len());
assert_eq!(
print_tree(db, &root_node, false, true),
print_tree(db, &node_from_token_stream, false, true)
);
}
#[test]
fn test_token_stream_expr_parser() {
let expr_code = indoc! {r#"
temp = a
"#};
let db = SimpleParserDatabase::default();
let token_stream = MockTokenStream {
tokens: vec![MockToken {
content: expr_code.to_string(),
span: TextSpan::from_str(expr_code),
}],
};
let (node_from_token_stream, _) = db.parse_token_stream_expr(&token_stream);
let node_text = node_from_token_stream.get_text(&db);
assert_eq!(node_text, expr_code);
}