lune/packages/cli/src/gen/selene_defs.rs

168 lines
6.2 KiB
Rust
Raw Normal View History

use anyhow::{Context, Result};
use full_moon::tokenizer::Symbol;
use serde_yaml::{Mapping as YamlMapping, Sequence as YamlSequence, Value as YamlValue};
use crate::gen::definitions::DefinitionsItemTag;
use super::definitions::{DefinitionsItem, DefinitionsItemKind, DefinitionsTree};
pub fn generate_from_type_definitions(contents: &str) -> Result<String> {
let tree = DefinitionsTree::from_type_definitions(contents)?;
let mut globals = YamlMapping::new();
2023-02-22 09:48:04 +00:00
let top_level_exported_items = tree.children().iter().filter(|top_level| {
top_level.is_exported()
&& (top_level.is_function()
|| top_level.children().iter().any(|top_level_child| {
top_level_child.is_tag() && top_level_child.get_name().unwrap() == "class"
}))
});
2023-02-22 09:48:04 +00:00
for top_level_item in top_level_exported_items {
match top_level_item.kind() {
DefinitionsItemKind::Table => {
let top_level_name = top_level_item
.get_name()
.context("Missing name for top-level doc item")?
.to_string();
for child_item in top_level_item
.children()
.iter()
2023-02-22 09:48:04 +00:00
.filter(|item| item.is_function() || item.is_table() || item.is_property())
{
let child_name = child_item
.get_name()
.context("Missing name for top-level child doc item")?
.to_string();
globals.insert(
YamlValue::String(format!("{top_level_name}.{child_name}")),
YamlValue::Mapping(doc_item_to_selene_yaml_mapping(child_item)?),
);
}
}
DefinitionsItemKind::Function => {
globals.insert(
YamlValue::String(
top_level_item
.get_name()
.context("Missing name for top-level doc item")?
.to_string(),
),
YamlValue::Mapping(doc_item_to_selene_yaml_mapping(top_level_item)?),
);
}
_ => unimplemented!("Globals other than tables and functions are not yet implemented"),
}
}
let mut contents = YamlMapping::new();
contents.insert(
YamlValue::String("globals".to_string()),
YamlValue::Mapping(globals),
);
Ok(format!(
"# Lune v{}\n---\n{}",
env!("CARGO_PKG_VERSION"),
serde_yaml::to_string(&contents).context("Failed to encode type definitions as yaml")?
))
}
fn doc_item_to_selene_yaml_mapping(item: &DefinitionsItem) -> Result<YamlMapping> {
let mut mapping = YamlMapping::new();
2023-02-22 09:48:04 +00:00
if item.is_property() || item.is_table() {
let property_access_tag = item
.children()
.iter()
.find_map(|child| {
if let Ok(tag) = DefinitionsItemTag::try_from(child) {
if tag.is_read_only() || tag.is_read_write() {
Some(tag)
} else {
None
}
} else {
None
}
})
.with_context(|| {
format!(
"Missing property access tag for doc item:\n{}",
item.get_name().unwrap()
)
})?;
mapping.insert(
YamlValue::String("property".to_string()),
YamlValue::String(
match property_access_tag {
DefinitionsItemTag::ReadOnly => "read-only",
DefinitionsItemTag::ReadWrite => "new-fields",
_ => unreachable!(),
}
.to_string(),
),
);
} else if item.is_function() {
let is_must_use = item.children().iter().any(|child| {
if let Ok(tag) = DefinitionsItemTag::try_from(child) {
tag.is_must_use()
} else {
false
}
});
if is_must_use {
mapping.insert(
YamlValue::String("must_use".to_string()),
YamlValue::Bool(true),
);
}
let mut args = YamlSequence::new();
for arg_type in item.arg_types() {
let mut arg_mapping = YamlMapping::new();
let (type_str, type_opt) = match arg_type.strip_suffix('?') {
Some(stripped) => (stripped, true),
None => (arg_type, false),
};
if type_opt {
arg_mapping.insert(
YamlValue::String("required".to_string()),
YamlValue::Bool(false),
);
}
arg_mapping.insert(
YamlValue::String("type".to_string()),
YamlValue::String(simplify_type_str_into_primitives(
type_str.trim_start_matches('(').trim_end_matches(')'),
)),
);
args.push(YamlValue::Mapping(arg_mapping));
}
mapping.insert(
YamlValue::String("args".to_string()),
YamlValue::Sequence(args),
);
}
Ok(mapping)
}
fn simplify_type_str_into_primitives(type_str: &str) -> String {
let separator = format!(" {} ", Symbol::Pipe);
// Simplify type strings even further into ones that selene can understand,
// turning types such as `{ bool }` or `"string-literal"` into `bool[]` and `string`
let mut primitives = Vec::new();
for type_inner in type_str.split(&separator) {
if type_inner.starts_with('{') && type_inner.ends_with('}') {
primitives.push(format!(
"{}[]",
type_inner
.trim_start_matches('{')
.trim_end_matches('}')
.trim()
));
} else if type_inner.starts_with('"') && type_inner.ends_with('"') {
primitives.push("string".to_string());
} else {
primitives.push(type_inner.to_string());
}
}
primitives.sort_unstable();
primitives.dedup();
primitives.join(&separator)
}