forked from TDAmeritrade/stumpy
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdocstring.py
executable file
·102 lines (86 loc) · 3.69 KB
/
docstring.py
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
#!/usr/bin/env python
import ast
import pathlib
import re
def get_docstring_args(fd, file_name, func_name, class_name=None):
"""
Extract docstring parameters from function definition
"""
docstring = ast.get_docstring(fd)
if len(re.findall(r"Parameters", docstring)) != 1:
msg = "Missing required 'Parameters' section in docstring in \n"
msg += f"file: {file_name}\n"
if class_name is not None:
msg += f"class: {class_name}\n"
msg += f"function/method: {func_name}\n"
raise RuntimeError(msg)
if class_name is None and len(re.findall(r"Returns", docstring)) != 1:
msg = "Missing required 'Returns' section in docstring in \n"
msg += f"file: {file_name}\n"
msg += f"function/method: {func_name}\n"
raise RuntimeError(msg)
if class_name is None:
params_section = re.findall(
r"(?<=Parameters)(.*)(?=Returns)", docstring, re.DOTALL
)[0]
else:
params_section = re.findall(r"(?<=Parameters)(.*)", docstring, re.DOTALL)[0]
args = re.findall(r"(\w+)\s+\:", params_section)
args = set([a for a in args if a != "i"]) # `i` should never be a parameter
return args
def get_signature_args(fd):
"""
Extract signature arguments from function definition
"""
return set([a.arg for a in fd.args.args if a.arg != "self"])
def check_args(doc_args, sig_args, file_name, func_name, class_name=None):
"""
Compare docstring arguments and signature argments
"""
diff_args = signature_args.difference(docstring_args)
if len(diff_args) > 0:
msg = "Found one or more arguments/parameters with missing docstring in \n"
msg += f"file: {file_name}\n"
if class_name is not None:
msg += f"class: {class_name}\n"
msg += f"function/method: {func_name}\n"
msg += f"parameter(s): {diff_args}\n"
raise RuntimeError(msg)
diff_args = docstring_args.difference(signature_args)
if len(diff_args) > 0:
msg = "Found one or more unsupported arguments/parameters with docstring in \n"
msg += f"file: {file_name}\n"
if class_name is not None:
msg += f"class: {class_name}\n"
msg += f"function/method: {func_name}\n"
msg += f"parameter(s): {diff_args}\n"
raise RuntimeError(msg)
ignore = ["__init__.py", "__pycache__"]
stumpy_path = pathlib.Path(__file__).parent / "stumpy"
filepaths = sorted(f for f in pathlib.Path(stumpy_path).iterdir() if f.is_file())
for filepath in filepaths:
if filepath.name not in ignore and str(filepath).endswith(".py"):
file_contents = ""
with open(filepath, encoding="utf8") as f:
file_contents = f.read()
module = ast.parse(file_contents)
# Check Functions
function_definitions = [
node for node in module.body if isinstance(node, ast.FunctionDef)
]
for fd in function_definitions:
docstring_args = get_docstring_args(fd, filepath.name, fd.name)
signature_args = get_signature_args(fd)
check_args(docstring_args, signature_args, filepath.name, fd.name)
# Check Class Methods
class_definitions = [
node for node in module.body if isinstance(node, ast.ClassDef)
]
for cd in class_definitions:
methods = [node for node in cd.body if isinstance(node, ast.FunctionDef)]
for fd in methods:
docstring_args = get_docstring_args(fd, filepath.name, fd.name, cd.name)
signature_args = get_signature_args(fd)
check_args(
docstring_args, signature_args, filepath.name, fd.name, cd.name
)