2023-02-21 20:06:11 +00:00
|
|
|
use anyhow::{Context, Result};
|
|
|
|
use full_moon::{
|
|
|
|
ast::{
|
2023-02-22 10:31:58 +00:00
|
|
|
types::{TypeFieldKey, TypeInfo},
|
2023-02-21 20:06:11 +00:00
|
|
|
Stmt,
|
|
|
|
},
|
|
|
|
tokenizer::{TokenReference, TokenType},
|
|
|
|
};
|
|
|
|
use regex::Regex;
|
|
|
|
|
2023-02-22 09:11:36 +00:00
|
|
|
use super::{
|
2023-02-22 10:31:58 +00:00
|
|
|
builder::DefinitionsItemBuilder, item::DefinitionsItem, moonwave::parse_moonwave_style_comment,
|
|
|
|
type_info_ext::TypeInfoExt,
|
2023-02-22 09:11:36 +00:00
|
|
|
};
|
|
|
|
|
2023-02-21 21:14:00 +00:00
|
|
|
#[derive(Debug, Clone)]
|
2023-02-22 09:53:00 +00:00
|
|
|
struct DefinitionsParserItem {
|
2023-02-21 20:06:11 +00:00
|
|
|
name: String,
|
|
|
|
comment: Option<String>,
|
|
|
|
type_info: TypeInfo,
|
|
|
|
}
|
|
|
|
|
2023-02-22 10:31:58 +00:00
|
|
|
#[derive(Debug, Default, Clone)]
|
|
|
|
pub struct DefinitionsParser {
|
|
|
|
found_top_level_items: Vec<DefinitionsParserItem>,
|
|
|
|
found_top_level_declares: Vec<String>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl DefinitionsParser {
|
|
|
|
pub fn new() -> Self {
|
|
|
|
Self {
|
|
|
|
..Default::default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn parse<S>(&mut self, contents: S) -> Result<()>
|
|
|
|
where
|
|
|
|
S: AsRef<str>,
|
|
|
|
{
|
|
|
|
// TODO: Properly handle the "declare class" syntax, for now we just skip it
|
|
|
|
let mut no_class_declares = contents.as_ref().to_string();
|
|
|
|
while let Some(dec) = no_class_declares.find("\ndeclare class") {
|
|
|
|
let end = no_class_declares.find("\nend").unwrap();
|
|
|
|
let before = &no_class_declares[0..dec];
|
|
|
|
let after = &no_class_declares[end + 4..];
|
|
|
|
no_class_declares = format!("{before}{after}");
|
|
|
|
}
|
|
|
|
// Replace declares with export type syntax that can be parsed by full_moon,
|
|
|
|
// find all declare statements and save declared names for later parsing
|
|
|
|
let regex_declare = Regex::new(r#"declare (\w+): "#).unwrap();
|
|
|
|
let resulting_contents = regex_declare
|
|
|
|
.replace_all(&no_class_declares, "export type $1 =")
|
|
|
|
.to_string();
|
|
|
|
let found_declares = regex_declare
|
|
|
|
.captures_iter(&no_class_declares)
|
|
|
|
.map(|cap| cap[1].to_string())
|
|
|
|
.collect();
|
|
|
|
// Parse contents into top-level parser items for later use
|
|
|
|
let mut found_top_level_items = Vec::new();
|
|
|
|
let ast =
|
|
|
|
full_moon::parse(&resulting_contents).context("Failed to parse type definitions")?;
|
|
|
|
for stmt in ast.nodes().stmts() {
|
|
|
|
if let Some((declaration, token_reference)) = match stmt {
|
|
|
|
Stmt::ExportedTypeDeclaration(exp) => {
|
|
|
|
Some((exp.type_declaration(), exp.export_token()))
|
|
|
|
}
|
|
|
|
Stmt::TypeDeclaration(typ) => Some((typ, typ.type_token())),
|
|
|
|
_ => None,
|
|
|
|
} {
|
|
|
|
found_top_level_items.push(DefinitionsParserItem {
|
|
|
|
name: declaration.type_name().token().to_string(),
|
|
|
|
comment: find_token_moonwave_comment(token_reference),
|
|
|
|
type_info: declaration.type_definition().clone(),
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Store results
|
|
|
|
self.found_top_level_items = found_top_level_items;
|
|
|
|
self.found_top_level_declares = found_declares;
|
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
|
|
|
fn convert_parser_item_into_doc_item(&self, item: DefinitionsParserItem) -> DefinitionsItem {
|
2023-02-22 09:53:00 +00:00
|
|
|
let mut builder = DefinitionsItemBuilder::new()
|
2023-02-22 10:31:58 +00:00
|
|
|
.with_kind(item.type_info.to_definitions_kind())
|
|
|
|
.with_name(&item.name);
|
|
|
|
if self.found_top_level_declares.contains(&item.name) {
|
2023-02-22 09:48:04 +00:00
|
|
|
builder = builder.as_exported();
|
|
|
|
}
|
2023-02-22 10:31:58 +00:00
|
|
|
if let Some(comment) = item.comment {
|
2023-02-21 20:06:11 +00:00
|
|
|
builder = builder.with_children(&parse_moonwave_style_comment(&comment));
|
|
|
|
}
|
2023-02-22 10:31:58 +00:00
|
|
|
if let Some(args) = item.type_info.extract_args_normalized() {
|
2023-02-22 09:11:36 +00:00
|
|
|
builder = builder.with_arg_types(&args);
|
|
|
|
}
|
2023-02-22 10:31:58 +00:00
|
|
|
if let TypeInfo::Table { fields, .. } = item.type_info {
|
2023-02-21 20:06:11 +00:00
|
|
|
for field in fields.iter() {
|
|
|
|
if let TypeFieldKey::Name(name) = field.key() {
|
2023-02-22 10:31:58 +00:00
|
|
|
builder = builder.with_child(self.convert_parser_item_into_doc_item(
|
|
|
|
DefinitionsParserItem {
|
2023-02-22 09:48:04 +00:00
|
|
|
name: name.token().to_string(),
|
|
|
|
comment: find_token_moonwave_comment(name),
|
|
|
|
type_info: field.value().clone(),
|
2023-02-22 10:31:58 +00:00
|
|
|
},
|
|
|
|
));
|
2023-02-21 20:06:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
builder.build().unwrap()
|
|
|
|
}
|
|
|
|
|
2023-02-22 10:31:58 +00:00
|
|
|
#[allow(clippy::unnecessary_wraps)]
|
|
|
|
pub fn into_definition_items(mut self) -> Result<Vec<DefinitionsItem>> {
|
|
|
|
let mut top_level_items = self.found_top_level_items.drain(..).collect::<Vec<_>>();
|
|
|
|
let results = top_level_items
|
|
|
|
.drain(..)
|
|
|
|
.map(|visitor_item| self.convert_parser_item_into_doc_item(visitor_item))
|
|
|
|
.collect();
|
|
|
|
Ok(results)
|
2023-02-21 20:06:11 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn find_token_moonwave_comment(token: &TokenReference) -> Option<String> {
|
|
|
|
token
|
|
|
|
.leading_trivia()
|
|
|
|
.filter_map(|trivia| match trivia.token_type() {
|
|
|
|
TokenType::MultiLineComment { blocks, comment } if blocks == &1 => Some(comment),
|
|
|
|
_ => None,
|
|
|
|
})
|
|
|
|
.last()
|
|
|
|
.map(|comment| comment.trim().to_string())
|
|
|
|
}
|