Skip to content

Commit

Permalink
Fix #7932
Browse files Browse the repository at this point in the history
Docgen should output attributes (#7932).

Output attributes (e.g., #[view] or #[resource_group(scope = global)])
from docgen for move code.  Currently only handles attibutes on
Function, Script/Module, and Struct/Resource.

Note that attributes are allowed on some other syntactic objects
(address scope, use statement, const definition, spec), but how to
format these usefully in docgen is unclear.  for these in docgen is
unclear.

Also fix an unrelated bug discovered when adapting
attribute_placement.move as a test of docgen: previously, only a
single Module/Script per input file would be output in docgen.
  • Loading branch information
brianrmurphy committed Jun 13, 2023
1 parent 58adba8 commit 37e8300
Show file tree
Hide file tree
Showing 5 changed files with 767 additions and 10 deletions.
96 changes: 86 additions & 10 deletions third_party/move/move-prover/move-docgen/src/docgen.rs
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ use log::{debug, info, warn};
use move_compiler::parser::keywords::{BUILTINS, CONTEXTUAL_KEYWORDS, KEYWORDS};
use move_core_types::account_address::AccountAddress;
use move_model::{
ast::{Address, ModuleName, SpecBlockInfo, SpecBlockTarget},
ast::{Address, Attribute, AttributeValue, ModuleName, SpecBlockInfo, SpecBlockTarget},
code_writer::{CodeWriter, CodeWriterLabel},
emit, emitln,
model::{
Expand Down Expand Up @@ -123,8 +123,8 @@ pub struct Docgen<'env> {
/// Mapping from module id to the set of schemas defined in this module.
/// We currently do not have this information in the environment.
declared_schemas: BTreeMap<ModuleId, BTreeSet<Symbol>>,
/// A list of file names and output generated for those files.
output: Vec<(String, String)>,
/// A map of file names to output generated for each file.
output: BTreeMap<String, String>,
/// Map from module id to information about this module.
infos: BTreeMap<ModuleId, ModuleInfo>,
/// Current code writer.
Expand Down Expand Up @@ -237,7 +237,15 @@ impl<'env> Docgen<'env> {
if !info.is_included && m.is_target() {
self.gen_module(&m, &info);
let path = self.make_file_in_out_dir(&info.target_file);
self.output.push((path, self.writer.extract_result()));
match self.output.get_mut(&path) {
Some(out) => {
out.push_str("\n\n");
out.push_str(&self.writer.extract_result());
},
None => {
self.output.insert(path, self.writer.extract_result());
},
}
}
}

Expand All @@ -250,7 +258,7 @@ impl<'env> Docgen<'env> {
{
let trimmed_content = content.trim();
if !trimmed_content.is_empty() {
for (_, out) in self.output.iter_mut() {
for out in self.output.values_mut() {
out.push_str("\n\n");
out.push_str(trimmed_content);
out.push('\n');
Expand All @@ -265,6 +273,9 @@ impl<'env> Docgen<'env> {
}

self.output
.iter()
.map(|(a, b)| (a.clone(), b.clone()))
.collect()
}

/// Compute the schemas declared in all modules. This information is currently not directly
Expand Down Expand Up @@ -372,10 +383,10 @@ impl<'env> Docgen<'env> {
}

// Add result to output.
self.output.push((
self.output.insert(
self.make_file_in_out_dir(output_file_name),
self.writer.extract_result(),
));
);
}

/// Compute ModuleInfo for all modules, considering root template content.
Expand Down Expand Up @@ -501,6 +512,65 @@ impl<'env> Docgen<'env> {
}
}

/// Get a readable version of an attribute.
fn gen_attribute(&self, attribute: &Attribute) -> String {
let annotation_body: String = match *attribute {
Attribute::Apply(ref _node_id, ref symbol, ref attribute_vector) => {
let symbol_string = self.name_string(*symbol).to_string();
if attribute_vector.is_empty() {
symbol_string
} else {
let value_string = self.gen_attributes_recursive(&attribute_vector[..]);
format!("{}({})", symbol_string, value_string)
}
},
Attribute::Assign(ref _node_id, ref symbol, ref attribute_value) => {
let symbol_string = self.name_string(*symbol).to_string();
match *attribute_value {
AttributeValue::Value(ref _node_id, ref value) => {
let value_string = self.env.display(value);
format!("{} = {}", symbol_string, value_string)
},
AttributeValue::Name(ref _node_id, ref module_name_option, ref symbol2) => {
let symbol2_name = self.name_string(*symbol2).to_string();
let module_prefix = match module_name_option {
None => "".to_string(),
Some(module_name) => {
format!("{}::", module_name.display_full(self.env))
},
};
format!("{} = {}{}", symbol_string, module_prefix, symbol2_name)
},
}
},
};
annotation_body
}

fn gen_attributes_recursive(&self, attributes: &[Attribute]) -> String {
if !attributes.is_empty() {
attributes
.iter()
.map(|attr| self.gen_attribute(attr))
.collect::<Vec<String>>()
.join(", ")
} else {
"".to_string()
}
}

fn gen_attributes(&self, attributes: &[Attribute]) -> String {
if !attributes.is_empty() {
attributes
.iter()
.map(|attr| format!("`#[{}]`<br>", self.gen_attribute(attr)))
.collect::<Vec<String>>()
.join("")
} else {
"".to_string()
}
}

/// Generates documentation for a module. The result is written into the current code
/// writer. Writer and other state is initialized if this module is standalone.
fn gen_module(&mut self, module_env: &ModuleEnv<'env>, info: &ModuleInfo) {
Expand All @@ -527,7 +597,8 @@ impl<'env> Docgen<'env> {
// Print header
self.section_header(
&format!(
"{} `{}`",
"{}{} `{}`",
self.gen_attributes(module_env.get_attributes()),
Self::module_modifier(module_env.get_name()),
module_env.get_name().display_full(module_env.env)
),
Expand Down Expand Up @@ -945,7 +1016,8 @@ impl<'env> Docgen<'env> {
// depends on the `is_resource()` predicate to add additional functions to structs declared
// with the `key` ability.
format!(
"{} `{}`",
"{}{} `{}`",
self.gen_attributes(struct_env.get_attributes()),
if struct_env.has_memory() {
"Resource"
} else {
Expand Down Expand Up @@ -1007,7 +1079,11 @@ impl<'env> Docgen<'env> {
let name = func_env.get_name();
if !is_script {
self.section_header(
&format!("Function `{}`", self.name_string(name)),
&format!(
"{}Function `{}`",
self.gen_attributes(func_env.get_attributes()),
self.name_string(name)
),
&self.label_for_module_item(&func_env.module_env, name),
);
self.increment_section_nest();
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,53 @@
#[attr1]
address 0x42 {
#[attr2]
#[attr7]
module M {
#[attr3]
use 0x42::N;

#[attr4]
struct S {}

#[attr4b]
#[resource_group(scope = global)]
struct T {}

#[attr2]
#[attr5]
const C: u64 = 0;

#[attr6]
#[resource_group_member(group = std::string::String)]
public fun foo() { N::bar() }

#[attr7]
spec foo {}
}
}

#[attr8]
module 0x42::N {
#[attr9]
friend 0x42::M;

#[attr10]
public fun bar() {}
}

#[attr11]
script {
#[attr12]
use 0x42::M;

#[attr13]
const C: u64 = 0;

#[attr14]
fun main() {
M::foo();
}

#[attr15]
spec main { }
}
Loading

0 comments on commit 37e8300

Please sign in to comment.