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
pub mod collection;
pub mod error;
use pyo3::prelude::*;
#[derive(Debug, PartialEq)]
pub struct FuncMetadata {
pub name: String,
pub is_coroutine: bool,
pub num_args: usize,
}
pub fn func_metadata(py: Python, func: &PyObject) -> PyResult<FuncMetadata> {
let name = func.getattr(py, "__name__")?.extract::<String>(py)?;
let is_coroutine = is_coroutine(py, func)?;
let inspect = py.import("inspect")?;
let args = inspect
.call_method1("getargs", (func.getattr(py, "__code__")?,))?
.getattr("args")?
.extract::<Vec<String>>()?;
Ok(FuncMetadata {
name,
is_coroutine,
num_args: args.len(),
})
}
fn is_coroutine(py: Python, func: &PyObject) -> PyResult<bool> {
let inspect = py.import("inspect")?;
inspect
.call_method1("iscoroutinefunction", (func,))?
.extract::<bool>()
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn function_metadata() -> PyResult<()> {
pyo3::prepare_freethreaded_python();
Python::with_gil(|py| {
let module = PyModule::from_code(
py,
r#"
def regular_func(first_arg, second_arg):
pass
async def async_func():
pass
"#,
"",
"",
)?;
let regular_func = module.getattr("regular_func")?.into_py(py);
assert_eq!(
FuncMetadata {
name: "regular_func".to_string(),
is_coroutine: false,
num_args: 2,
},
func_metadata(py, ®ular_func)?
);
let async_func = module.getattr("async_func")?.into_py(py);
assert_eq!(
FuncMetadata {
name: "async_func".to_string(),
is_coroutine: true,
num_args: 0,
},
func_metadata(py, &async_func)?
);
Ok(())
})
}
}