@@ -56,76 +56,128 @@ def parse_args():
56
56
57
57
def _check_params_in_description (rstfilename , paramstr ):
58
58
flag = True
59
- params_intitle = []
59
+ info = ""
60
+ params_in_title = []
60
61
if paramstr :
61
62
fake_func = ast .parse (f"def fake_func({ paramstr } ): pass" )
62
- for arg in fake_func .body [0 ].args .args :
63
- params_intitle .append (arg .arg )
63
+ # Iterate over all in_title parameters
64
+ num_defaults = len (fake_func .body [0 ].args .defaults )
65
+ num_args = len (fake_func .body [0 ].args .args )
66
+ # args & defaults
67
+ for i , arg in enumerate (fake_func .body [0 ].args .args ):
68
+ if i >= num_args - num_defaults :
69
+ default_value = fake_func .body [0 ].args .defaults [
70
+ i - (num_args - num_defaults )
71
+ ]
72
+ params_in_title .append (f"{ arg .arg } ={ default_value } " )
73
+ else :
74
+ params_in_title .append (arg .arg )
75
+ # posonlyargs
76
+ for arg in fake_func .body [0 ].args .posonlyargs :
77
+ params_in_title .append (arg .arg )
78
+ # vararg(*args)
79
+ if fake_func .body [0 ].args .vararg :
80
+ params_in_title .append (fake_func .body [0 ].args .vararg .arg )
81
+ # kwonlyargs & kw_defaults
82
+ for i , arg in enumerate (fake_func .body [0 ].args .kwonlyargs ):
83
+ if (
84
+ i < len (fake_func .body [0 ].args .kw_defaults )
85
+ and fake_func .body [0 ].args .kw_defaults [i ] is not None
86
+ ):
87
+ default_value = fake_func .body [0 ].args .kw_defaults [i ]
88
+ params_in_title .append (f"{ arg .arg } ={ default_value } " )
89
+ else :
90
+ params_in_title .append (arg .arg )
91
+ # **kwargs
92
+ if fake_func .body [0 ].args .kwarg :
93
+ params_in_title .append (fake_func .body [0 ].args .kwarg .arg )
64
94
65
95
funcdescnode = extract_params_desc_from_rst_file (rstfilename )
66
96
if funcdescnode :
67
97
items = funcdescnode .children [1 ].children [0 ].children
68
- if len (items ) != len (params_intitle ):
98
+ list_pat = r"^<list_item>.*</list_item>$"
99
+ if not re .match (list_pat , str (items [0 ])):
69
100
flag = False
101
+ info = "Something wrong with the format of params list in description, check it please."
102
+ elif len (items ) != len (params_in_title ):
103
+ flag = False
104
+ if not items :
105
+ info = (
106
+ "Params section in description is empty, check it please."
107
+ )
108
+ else :
109
+ info = f"The number of params in title does not match the params in description: { len (params_in_title )} != { len (items )} ."
70
110
print (f"check failed (parammeters description): { rstfilename } " )
71
111
else :
72
112
for i in range (len (items )):
73
- pname_intitle = params_intitle [i ].split ("=" )[0 ].strip ()
74
- mo = re .match (r"(\w+)\b.*" , items [i ].children [0 ].astext ())
113
+ pname_in_title = params_in_title [i ].split ("=" )[0 ].strip ()
114
+ mo = re .match (
115
+ r"\*{0,2}(\w+)\b.*" , items [i ].children [0 ].astext ()
116
+ )
75
117
if mo :
76
118
pname_indesc = mo .group (1 )
77
- if pname_indesc != pname_intitle :
119
+ if pname_indesc != pname_in_title :
78
120
flag = False
121
+ info = f"the following param in title does not match the param in description: { pname_in_title } != { pname_indesc } ."
79
122
print (
80
- f"check failed (parammeters description): { rstfilename } , { pname_indesc } != { pname_intitle } "
123
+ f"check failed (parammeters description): { rstfilename } , { pname_in_title } != { pname_indesc } "
81
124
)
82
125
else :
83
126
flag = False
127
+ info = f"param name '{ pname_in_title } ' not matched in description line{ i + 1 } , check it please."
84
128
print (
85
129
f"check failed (parammeters description): { rstfilename } , param name not found in { i } paragraph."
86
130
)
87
131
else :
88
- if params_intitle :
132
+ if params_in_title :
133
+ info = "params section not found in description, check it please."
89
134
print (
90
- f"check failed (parameters description not found): { rstfilename } , { params_intitle } ."
135
+ f"check failed (parameters description not found): { rstfilename } , { params_in_title } ."
91
136
)
92
137
flag = False
93
- return flag
138
+ return flag , info
94
139
95
140
96
141
def _check_params_in_description_with_fullargspec (rstfilename , funcname ):
97
142
flag = True
143
+ info = ""
98
144
funcspec = inspect .getfullargspec (eval (funcname ))
99
145
funcdescnode = extract_params_desc_from_rst_file (rstfilename )
100
146
if funcdescnode :
101
147
items = funcdescnode .children [1 ].children [0 ].children
102
148
params_inspec = funcspec .args
103
149
if len (items ) != len (params_inspec ):
104
150
flag = False
151
+ info = f"check_with_fullargspec failed (parammeters description): { rstfilename } "
105
152
print (f"check failed (parammeters description): { rstfilename } " )
106
153
else :
107
154
for i in range (len (items )):
108
- pname_intitle = params_inspec [i ]
109
- mo = re .match (r"(\w+)\b.*" , items [i ].children [0 ].astext ())
155
+ pname_in_title = params_inspec [i ]
156
+ mo = re .match (
157
+ r"\*{0,2}(\w+)\b.*" , items [i ].children [0 ].astext ()
158
+ )
110
159
if mo :
111
160
pname_indesc = mo .group (1 )
112
- if pname_indesc != pname_intitle :
161
+ if pname_indesc != pname_in_title :
113
162
flag = False
163
+ info = f"the following param in title does not match the param in description: { pname_in_title } != { pname_indesc } ."
114
164
print (
115
- f"check failed (parammeters description): { rstfilename } , { pname_indesc } != { pname_intitle } "
165
+ f"check failed (parammeters description): { rstfilename } , { pname_in_title } != { pname_indesc } "
116
166
)
117
167
else :
118
168
flag = False
169
+ info = f"param name '{ pname_in_title } ' not matched in description line{ i + 1 } , check it please."
119
170
print (
120
171
f"check failed (parammeters description): { rstfilename } , param name not found in { i } paragraph."
121
172
)
122
173
else :
123
174
if funcspec .args :
175
+ info = "params section not found in description, check it please."
124
176
print (
125
177
f"check failed (parameters description not found): { rstfilename } , { funcspec .args } ."
126
178
)
127
179
flag = False
128
- return flag
180
+ return flag , info
129
181
130
182
131
183
def check_api_parameters (rstfiles , apiinfo ):
@@ -138,11 +190,11 @@ def check_api_parameters(rstfiles, apiinfo):
138
190
2. Some COMPLICATED annotations may break the scripts.
139
191
"""
140
192
pat = re .compile (
141
- r"^\.\.\s+py:(method|function|class)::\s+(\S +)\s*\(\s*(.*)\s*\)\s*$"
193
+ r"^\.\.\s+py:(method|function|class)::\s+([^\s(] +)\s*(?: \(\s*(.*)\s*\))? \s*$"
142
194
)
143
195
check_passed = []
144
- check_failed = []
145
- api_notfound = []
196
+ check_failed = {}
197
+ api_notfound = {}
146
198
for rstfile in rstfiles :
147
199
rstfilename = osp .join ("../docs" , rstfile )
148
200
print (f"checking : { rstfile } " )
@@ -171,22 +223,24 @@ def check_api_parameters(rstfiles, apiinfo):
171
223
print (
172
224
f"check func:{ funcname } in { rstfilename } with { paramstr } "
173
225
)
174
- flag = _check_params_in_description (
226
+ flag , info = _check_params_in_description (
175
227
rstfilename , paramstr
176
228
)
177
229
else :
178
230
print (
179
231
f'check func:{ funcname } in { rstfilename } with { paramstr } , but different with json\' s { apiobj ["args" ]} '
180
232
)
181
- flag = _check_params_in_description (
233
+ flag , info = _check_params_in_description (
182
234
rstfilename , paramstr
183
235
)
184
236
else : # paddle.abs class_method does not have `args` in its json item.
185
237
print (
186
238
f"check func:{ funcname } in { rstfilename } with its FullArgSpec"
187
239
)
188
- flag = _check_params_in_description_with_fullargspec (
189
- rstfilename , funcname
240
+ flag , info = (
241
+ _check_params_in_description_with_fullargspec (
242
+ rstfilename , funcname
243
+ )
190
244
)
191
245
break
192
246
if not func_found_in_json : # may be inner functions
@@ -200,11 +254,12 @@ def check_api_parameters(rstfiles, apiinfo):
200
254
check_passed .append (rstfile )
201
255
print (f"check success: { rstfile } " )
202
256
else :
203
- check_failed . append ( rstfile )
257
+ check_failed [ rstfile ] = info
204
258
print (f"check failed: { rstfile } " )
205
259
break
206
260
if not func_found :
207
- api_notfound .append (rstfile )
261
+ info = 'funcname in title is not found, please check the format of ".. py:function::func()"'
262
+ api_notfound [rstfile ] = info
208
263
print (f"check failed (object not found): { rstfile } " )
209
264
print (f"checking done: { rstfile } " )
210
265
return check_passed , check_failed , api_notfound
@@ -220,9 +275,11 @@ def check_api_parameters(rstfiles, apiinfo):
220
275
result = True
221
276
if check_failed :
222
277
result = False
223
- print (f"check_api_parameters failed: { check_failed } " )
278
+ for path , info in check_failed .items ():
279
+ print (f"Checking failed file:{ path } \n Error:{ info } \n " )
224
280
if api_notfound :
225
- print (f"check_api_parameters funcname not found in: { api_notfound } " )
281
+ for path , info in api_notfound .items ():
282
+ print (f"Checking failed file:{ path } \n Error:{ info } \n " )
226
283
if result :
227
284
sys .exit (0 )
228
285
else :
0 commit comments