mirror of
https://github.com/lune-org/lune.git
synced 2024-12-12 13:00:37 +00:00
Use new typedefs parser to generate docs file
This commit is contained in:
parent
d6d0b74ce0
commit
8e894c7ac9
10 changed files with 239 additions and 436 deletions
|
@ -1,5 +1,3 @@
|
||||||
-- Lune v0.4.0
|
|
||||||
|
|
||||||
--[=[
|
--[=[
|
||||||
@class fs
|
@class fs
|
||||||
|
|
||||||
|
@ -527,5 +525,8 @@ declare warn: <T...>(T...) -> ()
|
||||||
|
|
||||||
--[=[
|
--[=[
|
||||||
Throws an error and prints a formatted version of it with a leading "[ERROR]" tag.
|
Throws an error and prints a formatted version of it with a leading "[ERROR]" tag.
|
||||||
|
|
||||||
|
@param message The error message to throw
|
||||||
|
@param level The stack level to throw the error at, defaults to 0
|
||||||
]=]
|
]=]
|
||||||
declare error: <T>(message: T, level: number?) -> ()
|
declare error: <T>(message: T, level: number?) -> ()
|
||||||
|
|
|
@ -8,8 +8,8 @@ use tokio::fs::{read_to_string, write};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
gen::{
|
gen::{
|
||||||
generate_docs_json_from_definitions, generate_selene_defs_from_definitions,
|
generate_docs_json_from_definitions, generate_luau_defs_from_definitions,
|
||||||
generate_wiki_dir_from_definitions,
|
generate_selene_defs_from_definitions, generate_wiki_dir_from_definitions,
|
||||||
},
|
},
|
||||||
utils::{
|
utils::{
|
||||||
files::find_parse_file_path,
|
files::find_parse_file_path,
|
||||||
|
@ -120,7 +120,7 @@ impl Cli {
|
||||||
if generate_file_requested {
|
if generate_file_requested {
|
||||||
if self.generate_luau_types {
|
if self.generate_luau_types {
|
||||||
generate_and_save_file(FILE_NAME_LUAU_TYPES, "Luau type definitions", || {
|
generate_and_save_file(FILE_NAME_LUAU_TYPES, "Luau type definitions", || {
|
||||||
Ok(FILE_CONTENTS_LUAU_TYPES.to_string())
|
generate_luau_defs_from_definitions(FILE_CONTENTS_LUAU_TYPES)
|
||||||
})
|
})
|
||||||
.await?;
|
.await?;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,52 +0,0 @@
|
||||||
use std::collections::HashMap;
|
|
||||||
|
|
||||||
use serde::{Deserialize, Serialize};
|
|
||||||
|
|
||||||
mod tag;
|
|
||||||
mod visitor;
|
|
||||||
|
|
||||||
pub use tag::*;
|
|
||||||
pub use visitor::*;
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Default, Debug, Clone)]
|
|
||||||
pub struct DocsGlobal {
|
|
||||||
pub documentation: String,
|
|
||||||
pub keys: HashMap<String, String>,
|
|
||||||
pub learn_more_link: String,
|
|
||||||
pub code_sample: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Default, Debug, Clone)]
|
|
||||||
pub struct DocsFunctionParamLink {
|
|
||||||
pub name: String,
|
|
||||||
pub documentation: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Default, Debug, Clone)]
|
|
||||||
pub struct DocsFunction {
|
|
||||||
#[serde(skip)]
|
|
||||||
pub global_name: String,
|
|
||||||
pub documentation: String,
|
|
||||||
pub params: Vec<DocsFunctionParamLink>,
|
|
||||||
pub returns: Vec<String>,
|
|
||||||
pub learn_more_link: String,
|
|
||||||
pub code_sample: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Default, Debug, Clone)]
|
|
||||||
pub struct DocsParam {
|
|
||||||
#[serde(skip)]
|
|
||||||
pub global_name: String,
|
|
||||||
#[serde(skip)]
|
|
||||||
pub function_name: String,
|
|
||||||
pub documentation: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Serialize, Deserialize, Default, Debug, Clone)]
|
|
||||||
pub struct DocsReturn {
|
|
||||||
#[serde(skip)]
|
|
||||||
pub global_name: String,
|
|
||||||
#[serde(skip)]
|
|
||||||
pub function_name: String,
|
|
||||||
pub documentation: String,
|
|
||||||
}
|
|
|
@ -1,66 +0,0 @@
|
||||||
use anyhow::{bail, Result};
|
|
||||||
|
|
||||||
#[derive(Clone, Debug, PartialEq, Eq)]
|
|
||||||
pub enum DocsTagKind {
|
|
||||||
Class,
|
|
||||||
Within,
|
|
||||||
Param,
|
|
||||||
Return,
|
|
||||||
Modifier,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DocsTagKind {
|
|
||||||
pub fn parse(s: &str) -> Result<Self> {
|
|
||||||
match s.trim().to_ascii_lowercase().as_ref() {
|
|
||||||
"class" => Ok(Self::Class),
|
|
||||||
"within" => Ok(Self::Within),
|
|
||||||
"param" => Ok(Self::Param),
|
|
||||||
"return" => Ok(Self::Return),
|
|
||||||
"must_use" | "read_only" | "new_fields" => Ok(Self::Modifier),
|
|
||||||
s => bail!("Unknown docs tag: '{}'", s),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
|
||||||
pub struct DocsTag {
|
|
||||||
pub kind: DocsTagKind,
|
|
||||||
pub name: String,
|
|
||||||
pub contents: String,
|
|
||||||
}
|
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
|
||||||
pub struct DocsTagList {
|
|
||||||
tags: Vec<DocsTag>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DocsTagList {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
Self { tags: vec![] }
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn push(&mut self, tag: DocsTag) {
|
|
||||||
self.tags.push(tag);
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn contains(&mut self, kind: DocsTagKind) -> bool {
|
|
||||||
self.tags.iter().any(|tag| tag.kind == kind)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn find(&mut self, kind: DocsTagKind) -> Option<&DocsTag> {
|
|
||||||
self.tags.iter().find(|tag| tag.kind == kind)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_empty(&self) -> bool {
|
|
||||||
self.tags.is_empty()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl IntoIterator for DocsTagList {
|
|
||||||
type Item = DocsTag;
|
|
||||||
type IntoIter = std::vec::IntoIter<Self::Item>;
|
|
||||||
|
|
||||||
fn into_iter(self) -> Self::IntoIter {
|
|
||||||
self.tags.into_iter()
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,216 +0,0 @@
|
||||||
use anyhow::Result;
|
|
||||||
use full_moon::{
|
|
||||||
ast::types::{ExportedTypeDeclaration, TypeField, TypeFieldKey},
|
|
||||||
parse as parse_luau_ast,
|
|
||||||
tokenizer::{Token, TokenType},
|
|
||||||
visitors::Visitor,
|
|
||||||
};
|
|
||||||
use regex::Regex;
|
|
||||||
|
|
||||||
use super::{
|
|
||||||
{DocsFunction, DocsFunctionParamLink, DocsGlobal, DocsParam, DocsReturn},
|
|
||||||
{DocsTag, DocsTagKind, DocsTagList},
|
|
||||||
};
|
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
|
||||||
pub struct DocumentationVisitor {
|
|
||||||
pub globals: Vec<(String, DocsGlobal)>,
|
|
||||||
pub functions: Vec<(String, DocsFunction)>,
|
|
||||||
pub params: Vec<(String, DocsParam)>,
|
|
||||||
pub returns: Vec<(String, DocsReturn)>,
|
|
||||||
tag_regex: Regex,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DocumentationVisitor {
|
|
||||||
pub fn new() -> Self {
|
|
||||||
let tag_regex = Regex::new(r#"^@(\S+)\s*(.*)$"#).unwrap();
|
|
||||||
Self {
|
|
||||||
globals: vec![],
|
|
||||||
functions: vec![],
|
|
||||||
params: vec![],
|
|
||||||
returns: vec![],
|
|
||||||
tag_regex,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn from_definitions(definitions_file_contents: &str) -> Result<Self> {
|
|
||||||
// TODO: Properly handle the "declare class" syntax, for now we just skip it
|
|
||||||
let mut no_declares = definitions_file_contents.to_string();
|
|
||||||
while let Some(dec) = no_declares.find("\ndeclare class") {
|
|
||||||
let end = no_declares.find("\nend").unwrap();
|
|
||||||
let before = &no_declares[0..dec];
|
|
||||||
let after = &no_declares[end + 4..];
|
|
||||||
no_declares = format!("{before}{after}");
|
|
||||||
}
|
|
||||||
let (regex, replacement) = (
|
|
||||||
Regex::new(r#"declare (?P<n>\w+): "#).unwrap(),
|
|
||||||
r#"export type $n = "#,
|
|
||||||
);
|
|
||||||
let defs_ast = parse_luau_ast(®ex.replace_all(&no_declares, replacement))?;
|
|
||||||
let mut visitor = DocumentationVisitor::new();
|
|
||||||
visitor.visit_ast(&defs_ast);
|
|
||||||
Ok(visitor)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_moonwave_style_tag(&self, line: &str) -> Option<DocsTag> {
|
|
||||||
if self.tag_regex.is_match(line) {
|
|
||||||
let captures = self.tag_regex.captures(line).unwrap();
|
|
||||||
let tag_kind = captures.get(1).unwrap().as_str();
|
|
||||||
let tag_rest = captures.get(2).unwrap().as_str();
|
|
||||||
let mut tag_words = tag_rest.split_whitespace().collect::<Vec<_>>();
|
|
||||||
let tag_name = if tag_words.is_empty() {
|
|
||||||
String::new()
|
|
||||||
} else {
|
|
||||||
tag_words.remove(0).to_string()
|
|
||||||
};
|
|
||||||
let tag_contents = tag_words.join(" ");
|
|
||||||
Some(DocsTag {
|
|
||||||
kind: DocsTagKind::parse(tag_kind).unwrap(),
|
|
||||||
name: tag_name,
|
|
||||||
contents: tag_contents,
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn parse_moonwave_style_comment(&self, comment: &str) -> (String, DocsTagList) {
|
|
||||||
let lines = comment.lines().map(str::trim).collect::<Vec<_>>();
|
|
||||||
let indent_len = lines.iter().fold(usize::MAX, |acc, line| {
|
|
||||||
let first = line.chars().enumerate().find_map(|(idx, ch)| {
|
|
||||||
if ch.is_alphanumeric() {
|
|
||||||
Some(idx)
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
});
|
|
||||||
if let Some(first_alphanumeric) = first {
|
|
||||||
if first_alphanumeric > 0 {
|
|
||||||
acc.min(first_alphanumeric - 1)
|
|
||||||
} else {
|
|
||||||
0
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
acc
|
|
||||||
}
|
|
||||||
});
|
|
||||||
let unindented_lines = lines.iter().map(|line| &line[indent_len..]);
|
|
||||||
let mut doc_lines = Vec::new();
|
|
||||||
let mut doc_tags = DocsTagList::new();
|
|
||||||
for line in unindented_lines {
|
|
||||||
if let Some(tag) = self.parse_moonwave_style_tag(line) {
|
|
||||||
doc_tags.push(tag);
|
|
||||||
} else {
|
|
||||||
doc_lines.push(line);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
(doc_lines.join("\n").trim().to_owned(), doc_tags)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn extract_moonwave_comment(&mut self, token: &Token) -> Option<(String, DocsTagList)> {
|
|
||||||
if let TokenType::MultiLineComment { comment, .. } = token.token_type() {
|
|
||||||
let (doc, tags) = self.parse_moonwave_style_comment(comment);
|
|
||||||
if doc.is_empty() && tags.is_empty() {
|
|
||||||
None
|
|
||||||
} else {
|
|
||||||
Some((doc, tags))
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Visitor for DocumentationVisitor {
|
|
||||||
fn visit_exported_type_declaration(&mut self, node: &ExportedTypeDeclaration) {
|
|
||||||
for token in node.export_token().leading_trivia() {
|
|
||||||
if let Some((doc, mut tags)) = self.extract_moonwave_comment(token) {
|
|
||||||
if tags.contains(DocsTagKind::Class) {
|
|
||||||
self.globals.push((
|
|
||||||
node.type_declaration().type_name().token().to_string(),
|
|
||||||
DocsGlobal {
|
|
||||||
documentation: doc,
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn visit_type_field(&mut self, node: &TypeField) {
|
|
||||||
// Parse out names, moonwave comments from the ast
|
|
||||||
let mut parsed_data = Vec::new();
|
|
||||||
if let TypeFieldKey::Name(name) = node.key() {
|
|
||||||
for token in name.leading_trivia() {
|
|
||||||
if let Some((doc, mut tags)) = self.extract_moonwave_comment(token) {
|
|
||||||
if let Some(within) = tags.find(DocsTagKind::Within).map(ToOwned::to_owned) {
|
|
||||||
parsed_data.push((within.name, name, doc, tags));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (global_name, name, doc, tags) in parsed_data {
|
|
||||||
// Find the global definition, which is guaranteed to
|
|
||||||
// be visited and parsed before its inner members, and
|
|
||||||
// add a ref to the found function / member to it
|
|
||||||
let name = name.token().to_string();
|
|
||||||
for (name, global) in &mut self.globals {
|
|
||||||
if name == &global_name {
|
|
||||||
global.keys.insert(name.clone(), name.clone());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Look through tags to find and create doc params and returns
|
|
||||||
let mut param_links = Vec::new();
|
|
||||||
let mut return_links = Vec::new();
|
|
||||||
for tag in tags {
|
|
||||||
match tag.kind {
|
|
||||||
DocsTagKind::Param => {
|
|
||||||
let idx_string = param_links.len().to_string();
|
|
||||||
self.params.push((
|
|
||||||
idx_string.clone(),
|
|
||||||
DocsParam {
|
|
||||||
global_name: global_name.clone(),
|
|
||||||
function_name: name.clone(),
|
|
||||||
documentation: tag.contents.trim().to_owned(),
|
|
||||||
},
|
|
||||||
));
|
|
||||||
param_links.push(DocsFunctionParamLink {
|
|
||||||
name: tag.name.clone(),
|
|
||||||
documentation: idx_string.clone(),
|
|
||||||
});
|
|
||||||
}
|
|
||||||
DocsTagKind::Return => {
|
|
||||||
// NOTE: Returns don't have names but we still parse
|
|
||||||
// them as such, so we should concat name & contents
|
|
||||||
let doc = format!("{} {}", tag.name.trim(), tag.contents.trim());
|
|
||||||
let idx_string = return_links.len().to_string();
|
|
||||||
self.returns.push((
|
|
||||||
idx_string.clone(),
|
|
||||||
DocsReturn {
|
|
||||||
global_name: global_name.clone(),
|
|
||||||
function_name: name.clone(),
|
|
||||||
documentation: doc,
|
|
||||||
},
|
|
||||||
));
|
|
||||||
return_links.push(idx_string.clone());
|
|
||||||
}
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Finally, add our complete doc
|
|
||||||
// function with links into the list
|
|
||||||
self.functions.push((
|
|
||||||
name,
|
|
||||||
DocsFunction {
|
|
||||||
global_name,
|
|
||||||
documentation: doc,
|
|
||||||
params: param_links,
|
|
||||||
returns: return_links,
|
|
||||||
..Default::default()
|
|
||||||
},
|
|
||||||
));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,65 +1,228 @@
|
||||||
use std::collections::HashMap;
|
|
||||||
|
|
||||||
use anyhow::{Context, Result};
|
use anyhow::{Context, Result};
|
||||||
|
use lazy_static::lazy_static;
|
||||||
use serde_json::{Map as JsonMap, Value as JsonValue};
|
use serde_json::{Map as JsonMap, Value as JsonValue};
|
||||||
|
|
||||||
use super::doc::{DocsFunctionParamLink, DocumentationVisitor};
|
use super::definitions::{DefinitionsItem, DefinitionsItemTag, DefinitionsTree};
|
||||||
|
|
||||||
|
lazy_static! {
|
||||||
|
static ref KEY_DOCUMENTATION: String = "documentation".to_string();
|
||||||
|
static ref KEY_KEYS: String = "keys".to_string();
|
||||||
|
static ref KEY_NAME: String = "name".to_string();
|
||||||
|
static ref KEY_CODE_SAMPLE: String = "code_sample".to_string();
|
||||||
|
static ref KEY_LEARN_MORE_LINK: String = "learn_more_link".to_string();
|
||||||
|
static ref VALUE_EMPTY: String = String::new();
|
||||||
|
}
|
||||||
|
|
||||||
pub fn generate_from_type_definitions(contents: &str, namespace: &str) -> Result<String> {
|
pub fn generate_from_type_definitions(contents: &str, namespace: &str) -> Result<String> {
|
||||||
let visitor = DocumentationVisitor::from_definitions(contents)?;
|
let tree = DefinitionsTree::from_type_definitions(contents)?;
|
||||||
/*
|
/*
|
||||||
Extract globals, functions, params, returns from the visitor
|
Extract globals, functions, params, returns from the type definitions tree
|
||||||
Here we will also convert the plain names into proper namespaced names according to the spec at
|
Here we will also convert the plain names into proper namespaced names according to the spec at
|
||||||
https://raw.githubusercontent.com/MaximumADHD/Roblox-Client-Tracker/roblox/api-docs/en-us.json
|
https://raw.githubusercontent.com/MaximumADHD/Roblox-Client-Tracker/roblox/api-docs/en-us.json
|
||||||
*/
|
*/
|
||||||
let mut map = JsonMap::new();
|
let mut map = JsonMap::new();
|
||||||
for (name, mut doc) in visitor.globals {
|
// Go over all the exported classes first (globals)
|
||||||
doc.keys = doc
|
let exported_items = tree.children().iter().filter(|item| {
|
||||||
.keys
|
item.is_exported()
|
||||||
.iter()
|
&& (item.is_function()
|
||||||
.map(|(key, value)| (key.clone(), format!("@{namespace}/{name}.{value}")))
|
|| item.children().iter().any(|item_child| {
|
||||||
.collect::<HashMap<String, String>>();
|
item_child.is_tag() && item_child.get_name().unwrap() == "class"
|
||||||
map.insert(format!("@{namespace}/{name}"), serde_json::to_value(doc)?);
|
}))
|
||||||
}
|
});
|
||||||
for (name, mut doc) in visitor.functions {
|
for item in exported_items {
|
||||||
doc.params = doc
|
parse_and_insert(&mut map, item, namespace, None)?;
|
||||||
.params
|
|
||||||
.iter()
|
|
||||||
.map(|param| DocsFunctionParamLink {
|
|
||||||
name: param.name.clone(),
|
|
||||||
documentation: format!(
|
|
||||||
"@{namespace}/{}.{name}/param/{}",
|
|
||||||
doc.global_name, param.documentation
|
|
||||||
),
|
|
||||||
})
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
doc.returns = doc
|
|
||||||
.returns
|
|
||||||
.iter()
|
|
||||||
.map(|ret| format!("@{namespace}/{}.{name}/return/{ret}", doc.global_name))
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
map.insert(
|
|
||||||
format!("@{namespace}/{}.{name}", doc.global_name),
|
|
||||||
serde_json::to_value(doc)?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
for (name, doc) in visitor.params {
|
|
||||||
map.insert(
|
|
||||||
format!(
|
|
||||||
"@{namespace}/{}.{}/param/{name}",
|
|
||||||
doc.global_name, doc.function_name
|
|
||||||
),
|
|
||||||
serde_json::to_value(doc)?,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
for (name, doc) in visitor.returns {
|
|
||||||
map.insert(
|
|
||||||
format!(
|
|
||||||
"@{namespace}/{}.{}/return/{name}",
|
|
||||||
doc.global_name, doc.function_name
|
|
||||||
),
|
|
||||||
serde_json::to_value(doc)?,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
// Go over the rest, these will be global types
|
||||||
|
// that exported items are referencing somewhere
|
||||||
serde_json::to_string_pretty(&JsonValue::Object(map)).context("Failed to encode docs as json")
|
serde_json::to_string_pretty(&JsonValue::Object(map)).context("Failed to encode docs as json")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[allow(clippy::too_many_lines)]
|
||||||
|
fn parse_and_insert(
|
||||||
|
map: &mut JsonMap<String, JsonValue>,
|
||||||
|
item: &DefinitionsItem,
|
||||||
|
namespace: &str,
|
||||||
|
parent: Option<&DefinitionsItem>,
|
||||||
|
) -> Result<()> {
|
||||||
|
let mut item_map = JsonMap::new();
|
||||||
|
let item_name = item
|
||||||
|
.get_name()
|
||||||
|
.with_context(|| format!("Missing name for doc item: {item:#?}"))?;
|
||||||
|
// Include parent name in full name, unless there is no parent (top-level global)
|
||||||
|
let item_name_full = match parent {
|
||||||
|
Some(parent) => format!(
|
||||||
|
"{}.{item_name}",
|
||||||
|
parent
|
||||||
|
.get_name()
|
||||||
|
.with_context(|| format!("Missing parent name for doc item: {item:#?}"))?
|
||||||
|
),
|
||||||
|
None => item_name.to_string(),
|
||||||
|
};
|
||||||
|
// Try to parse params & returns to use later
|
||||||
|
let mut params = Vec::new();
|
||||||
|
let mut returns = Vec::new();
|
||||||
|
if item.is_function() {
|
||||||
|
// Map and separate found tags into params & returns
|
||||||
|
let mut tags = item
|
||||||
|
.children()
|
||||||
|
.iter()
|
||||||
|
.filter_map(|child| {
|
||||||
|
if let Ok(tag) = DefinitionsItemTag::try_from(child) {
|
||||||
|
Some(tag)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
for tag in tags.drain(..) {
|
||||||
|
if tag.is_param() {
|
||||||
|
params.push(tag);
|
||||||
|
} else if tag.is_return() {
|
||||||
|
returns.push(tag);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Try to parse the description for this typedef item, if it has one,
|
||||||
|
// insert description + code sample + learn more link if they exist
|
||||||
|
if let Some(description) = item.children().iter().find(|child| child.is_description()) {
|
||||||
|
let (description, code_sample, learn_more_link) = try_parse_description_for_docs(
|
||||||
|
description
|
||||||
|
.get_value()
|
||||||
|
.context("Missing description value for doc item")?
|
||||||
|
.to_string(),
|
||||||
|
);
|
||||||
|
item_map.insert(KEY_DOCUMENTATION.clone(), JsonValue::String(description));
|
||||||
|
if let Some(code_sample) = code_sample {
|
||||||
|
item_map.insert(KEY_CODE_SAMPLE.clone(), JsonValue::String(code_sample));
|
||||||
|
} else {
|
||||||
|
item_map.insert(
|
||||||
|
KEY_CODE_SAMPLE.clone(),
|
||||||
|
JsonValue::String(VALUE_EMPTY.clone()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if let Some(learn_more_link) = learn_more_link {
|
||||||
|
item_map.insert(
|
||||||
|
KEY_LEARN_MORE_LINK.clone(),
|
||||||
|
JsonValue::String(learn_more_link),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
item_map.insert(
|
||||||
|
KEY_LEARN_MORE_LINK.clone(),
|
||||||
|
JsonValue::String(VALUE_EMPTY.clone()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
If the typedef item is a table, we should include keys
|
||||||
|
which are references from this global to its members,
|
||||||
|
then we should parse its members and add them in
|
||||||
|
|
||||||
|
If it is a function, we should parse its params and args,
|
||||||
|
make links to them in this object, and then add them in as
|
||||||
|
separate items into the globals map, with their documentation
|
||||||
|
*/
|
||||||
|
if item.is_table() {
|
||||||
|
let mut keys = item
|
||||||
|
.children()
|
||||||
|
.iter()
|
||||||
|
.filter_map(|child| {
|
||||||
|
if child.is_property() || child.is_table() || child.is_function() {
|
||||||
|
Some(child.get_name().expect("Missing name for doc item child"))
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
if keys.is_empty() {
|
||||||
|
item_map.insert("keys".to_string(), JsonValue::Object(JsonMap::new()));
|
||||||
|
} else {
|
||||||
|
let mut keys_map = JsonMap::new();
|
||||||
|
for key in keys.drain(..) {
|
||||||
|
keys_map.insert(
|
||||||
|
key.to_string(),
|
||||||
|
JsonValue::String(format!("@{namespace}/{item_name_full}.{key}")),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
item_map.insert("keys".to_string(), JsonValue::Object(keys_map));
|
||||||
|
}
|
||||||
|
} else if item.is_function() {
|
||||||
|
// Add links to params
|
||||||
|
if params.is_empty() {
|
||||||
|
item_map.insert("params".to_string(), JsonValue::Array(vec![]));
|
||||||
|
} else {
|
||||||
|
let mut params_vec = Vec::new();
|
||||||
|
for (index, param) in params.iter().enumerate() {
|
||||||
|
let mut param_map = JsonMap::new();
|
||||||
|
if let DefinitionsItemTag::Param((name, _)) = param {
|
||||||
|
param_map.insert(KEY_NAME.clone(), JsonValue::String(name.to_string()));
|
||||||
|
param_map.insert(
|
||||||
|
KEY_DOCUMENTATION.clone(),
|
||||||
|
JsonValue::String(format!("@{namespace}/{item_name_full}/param/{index}")),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
params_vec.push(JsonValue::Object(param_map));
|
||||||
|
}
|
||||||
|
item_map.insert("params".to_string(), JsonValue::Array(params_vec));
|
||||||
|
}
|
||||||
|
// Add links to returns
|
||||||
|
if returns.is_empty() {
|
||||||
|
item_map.insert("returns".to_string(), JsonValue::Array(vec![]));
|
||||||
|
} else {
|
||||||
|
let mut returns_vec = Vec::new();
|
||||||
|
for (index, _) in returns.iter().enumerate() {
|
||||||
|
returns_vec.push(JsonValue::String(format!(
|
||||||
|
"@{namespace}/{item_name_full}/return/{index}"
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
item_map.insert("returns".to_string(), JsonValue::Array(returns_vec));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
map.insert(
|
||||||
|
format!("@{namespace}/{item_name_full}"),
|
||||||
|
JsonValue::Object(item_map),
|
||||||
|
);
|
||||||
|
if item.is_table() {
|
||||||
|
for child in item
|
||||||
|
.children()
|
||||||
|
.iter()
|
||||||
|
.filter(|child| !child.is_description() && !child.is_tag())
|
||||||
|
{
|
||||||
|
parse_and_insert(map, child, namespace, Some(item))?;
|
||||||
|
}
|
||||||
|
} else if item.is_function() {
|
||||||
|
// FIXME: It seems the order of params and returns here is not
|
||||||
|
// deterministic, they can be unordered which leads to confusing docs
|
||||||
|
for (index, param) in params.iter().enumerate() {
|
||||||
|
let mut param_map = JsonMap::new();
|
||||||
|
if let DefinitionsItemTag::Param((_, doc)) = param {
|
||||||
|
param_map.insert(
|
||||||
|
KEY_DOCUMENTATION.clone(),
|
||||||
|
JsonValue::String(format!("{doc}\n\n---\n")),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
map.insert(
|
||||||
|
format!("@{namespace}/{item_name_full}/param/{index}"),
|
||||||
|
JsonValue::Object(param_map),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
for (index, ret) in returns.iter().enumerate() {
|
||||||
|
let mut return_map = JsonMap::new();
|
||||||
|
if let DefinitionsItemTag::Return(doc) = ret {
|
||||||
|
return_map.insert(
|
||||||
|
KEY_DOCUMENTATION.clone(),
|
||||||
|
JsonValue::String(doc.to_string()),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
map.insert(
|
||||||
|
format!("@{namespace}/{item_name_full}/return/{index}"),
|
||||||
|
JsonValue::Object(return_map),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn try_parse_description_for_docs(description: String) -> (String, Option<String>, Option<String>) {
|
||||||
|
// TODO: Implement this
|
||||||
|
(description, None, None)
|
||||||
|
}
|
||||||
|
|
10
packages/cli/src/gen/luau_defs.rs
Normal file
10
packages/cli/src/gen/luau_defs.rs
Normal file
|
@ -0,0 +1,10 @@
|
||||||
|
use anyhow::Result;
|
||||||
|
|
||||||
|
#[allow(clippy::unnecessary_wraps)]
|
||||||
|
pub fn generate_from_type_definitions(contents: &str) -> Result<String> {
|
||||||
|
Ok(format!(
|
||||||
|
"--> Lune v{}\n\n{}",
|
||||||
|
env!("CARGO_PKG_VERSION"),
|
||||||
|
contents
|
||||||
|
))
|
||||||
|
}
|
|
@ -1,14 +1,11 @@
|
||||||
mod doc;
|
|
||||||
mod docs_file;
|
mod docs_file;
|
||||||
|
mod luau_defs;
|
||||||
mod selene_defs;
|
mod selene_defs;
|
||||||
mod wiki_dir;
|
mod wiki_dir;
|
||||||
|
|
||||||
pub mod definitions;
|
pub mod definitions;
|
||||||
|
|
||||||
pub use docs_file::generate_from_type_definitions as generate_docs_json_from_definitions;
|
pub use docs_file::generate_from_type_definitions as generate_docs_json_from_definitions;
|
||||||
|
pub use luau_defs::generate_from_type_definitions as generate_luau_defs_from_definitions;
|
||||||
pub use selene_defs::generate_from_type_definitions as generate_selene_defs_from_definitions;
|
pub use selene_defs::generate_from_type_definitions as generate_selene_defs_from_definitions;
|
||||||
pub use wiki_dir::generate_from_type_definitions as generate_wiki_dir_from_definitions;
|
pub use wiki_dir::generate_from_type_definitions as generate_wiki_dir_from_definitions;
|
||||||
|
|
||||||
pub use self::doc::DocumentationVisitor;
|
|
||||||
|
|
||||||
pub const GENERATED_COMMENT_TAG: &str = "@generated with lune-cli";
|
|
||||||
|
|
|
@ -4,11 +4,12 @@ use anyhow::{Context, Result};
|
||||||
|
|
||||||
use tokio::fs::{create_dir_all, write};
|
use tokio::fs::{create_dir_all, write};
|
||||||
|
|
||||||
use super::doc::DocumentationVisitor;
|
use super::definitions::DefinitionsTree;
|
||||||
use super::GENERATED_COMMENT_TAG;
|
|
||||||
|
pub const GENERATED_COMMENT_TAG: &str = "@generated with lune-cli";
|
||||||
|
|
||||||
pub async fn generate_from_type_definitions(contents: &str) -> Result<()> {
|
pub async fn generate_from_type_definitions(contents: &str) -> Result<()> {
|
||||||
let visitor = DocumentationVisitor::from_definitions(contents)?;
|
let tree = DefinitionsTree::from_type_definitions(contents)?;
|
||||||
// Create the wiki dir at the repo root
|
// Create the wiki dir at the repo root
|
||||||
let root = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
let root = PathBuf::from(env!("CARGO_MANIFEST_DIR"))
|
||||||
.join("../../")
|
.join("../../")
|
||||||
|
@ -17,40 +18,5 @@ pub async fn generate_from_type_definitions(contents: &str) -> Result<()> {
|
||||||
create_dir_all(&root.join("wiki"))
|
create_dir_all(&root.join("wiki"))
|
||||||
.await
|
.await
|
||||||
.context("Failed to create wiki dir")?;
|
.context("Failed to create wiki dir")?;
|
||||||
for global in &visitor.globals {
|
|
||||||
// Create the dir for this global
|
|
||||||
let global_dir_path = root.join("wiki").join("api-reference").join(&global.0);
|
|
||||||
create_dir_all(&global_dir_path)
|
|
||||||
.await
|
|
||||||
.context("Failed to create doc dir for global")?;
|
|
||||||
// Create the markdown docs file for this global
|
|
||||||
let mut contents = String::new();
|
|
||||||
writeln!(contents, "<!-- {GENERATED_COMMENT_TAG} -->\n")?;
|
|
||||||
writeln!(contents, "# **{}**\n", global.0)?;
|
|
||||||
writeln!(contents, "{}\n", global.1.documentation)?;
|
|
||||||
if !global.1.code_sample.is_empty() {
|
|
||||||
writeln!(contents, "{}", global.1.code_sample)?;
|
|
||||||
}
|
|
||||||
let funcs = visitor
|
|
||||||
.functions
|
|
||||||
.iter()
|
|
||||||
.filter(|f| f.1.global_name == global.0)
|
|
||||||
.collect::<Vec<_>>();
|
|
||||||
if !funcs.is_empty() {
|
|
||||||
writeln!(contents, "## Functions\n")?;
|
|
||||||
for func in funcs {
|
|
||||||
writeln!(contents, "### {}\n", func.0)?;
|
|
||||||
writeln!(contents, "{}\n", func.1.documentation)?;
|
|
||||||
if !func.1.code_sample.is_empty() {
|
|
||||||
writeln!(contents, "{}", func.1.code_sample)?;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Write the file in the dir, with the same
|
|
||||||
// name as the dir to create an "index" page
|
|
||||||
write(&global_dir_path.join(format!("{}.md", &global.0)), contents)
|
|
||||||
.await
|
|
||||||
.context("Failed to create doc file for global")?;
|
|
||||||
}
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,7 @@
|
||||||
use serde_json::Value as JsonValue;
|
use serde_json::Value as JsonValue;
|
||||||
use serde_yaml::Value as YamlValue;
|
use serde_yaml::Value as YamlValue;
|
||||||
|
|
||||||
use crate::gen::DocumentationVisitor;
|
use crate::gen::definitions::DefinitionsTree;
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
pub enum FileType {
|
pub enum FileType {
|
||||||
|
@ -16,7 +16,7 @@ impl FileType {
|
||||||
Some(Self::Json)
|
Some(Self::Json)
|
||||||
} else if serde_yaml::from_str::<YamlValue>(contents).is_ok() {
|
} else if serde_yaml::from_str::<YamlValue>(contents).is_ok() {
|
||||||
Some(Self::Yaml)
|
Some(Self::Yaml)
|
||||||
} else if DocumentationVisitor::from_definitions(contents).is_ok() {
|
} else if DefinitionsTree::from_type_definitions(contents).is_ok() {
|
||||||
Some(Self::Luau)
|
Some(Self::Luau)
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
|
|
Loading…
Reference in a new issue