1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
use anyhow::*;
use docstring::normalize_doc_string;
use errmapgen::{ErrorDescription, ErrorMapping};
use move_bytecode_verifier::script_signature;
use move_core_types::identifier::IdentStr;
use move_idl_types::*;
use move_model::{
model::{GlobalEnv, ModuleEnv},
ty,
};
use std::collections::BTreeMap;
use move_idl_types::IDLError;
use crate::convert::get_type_tag_for_type;
use super::{gen_function::generate_idl_for_function, gen_struct::generate_idl_for_struct};
pub fn generate_idl_for_module(
env: &GlobalEnv,
error_mapping: &ErrorMapping,
module_env: ModuleEnv,
) -> anyhow::Result<IDLModule> {
let script_iter: Vec<_> = module_env
.get_functions()
.filter(|func| {
let module = module_env.get_verified_module();
let func_name = module_env.symbol_pool().string(func.get_name());
let func_ident = IdentStr::new(&func_name).unwrap();
func.is_entry()
&& script_signature::verify_module_function_signature_by_name(
module,
func_ident,
script_signature::no_additional_script_signature_checks,
)
.is_ok()
&& func
.get_parameters()
.iter()
.skip_while(|param| match ¶m.1 {
ty::Type::Primitive(ty::PrimitiveType::Signer) => true,
ty::Type::Reference(_, inner) => {
matches!(&**inner, ty::Type::Primitive(ty::PrimitiveType::Signer))
}
_ => false,
})
.all(|param| {
matches!(
get_type_tag_for_type(¶m.1),
Err(_) | Result::Ok(Some(_))
)
})
&& func.get_return_count() == 0
})
.collect();
let structs = generate_idl_structs_for_module(env, module_env.clone())?;
let mut functions = Vec::new();
for func in &script_iter {
functions.push(generate_idl_for_function(env, module_env.clone(), func)?);
}
let module_id = module_env.get_verified_module().self_id().into();
Ok(IDLModule {
module_id,
doc: normalize_doc_string(module_env.get_doc()),
functions,
structs,
errors: generate_idl_errors_for_module(error_mapping, &module_env)?,
})
}
pub fn generate_idl_structs_for_module(
env: &GlobalEnv,
module_env: ModuleEnv,
) -> Result<Vec<IDLStruct>> {
let mut ret = vec![];
for struct_env in module_env.into_structs() {
let idl_struct = generate_idl_for_struct(env, &struct_env)?;
ret.push(idl_struct)
}
Ok(ret)
}
pub fn generate_idl_errors_for_module(
error_mapping: &ErrorMapping,
module_env: &ModuleEnv,
) -> Result<BTreeMap<u64, IDLError>> {
let module_id: ModuleIdData = module_env.get_verified_module().self_id().into();
let result: Option<&BTreeMap<u64, ErrorDescription>> =
error_mapping.module_error_maps.get(&module_id);
Ok(match result {
Some(result) => result
.iter()
.map(|(k, v)| -> (u64, IDLError) { (*k, v.clone().into()) })
.collect(),
None => BTreeMap::new(),
})
}