Fri, 02 Apr 2021 15:35:44 +0200
Code Style Checker
- continued to implement checkers for potential code simplifications
8189 | 1 | # -*- coding: utf-8 -*- |
2 | ||
3 | # Copyright (c) 2021 Detlev Offenbach <detlev@die-offenbachs.de> | |
4 | # | |
5 | ||
6 | """ | |
7 | Module implementing a node visitor checking for code that could be simplified. | |
8 | """ | |
9 | ||
10 | import ast | |
11 | import collections | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
12 | import itertools |
8189 | 13 | |
14 | try: | |
15 | from ast import unparse | |
16 | except ImportError: | |
17 | # Python < 3.9 | |
18 | from .ast_unparse import unparse | |
19 | ||
20 | ###################################################################### | |
21 | ## The following code is derived from the flake8-simplify package. | |
22 | ## | |
23 | ## Original License: | |
24 | ## | |
25 | ## MIT License | |
26 | ## | |
27 | ## Copyright (c) 2020 Martin Thoma | |
28 | ## | |
29 | ## Permission is hereby granted, free of charge, to any person obtaining a copy | |
30 | ## of this software and associated documentation files (the "Software"), to | |
31 | ## deal in the Software without restriction, including without limitation the | |
32 | ## rights to use, copy, modify, merge, publish, distribute, sublicense, and/or | |
33 | ## sell copies of the Software, and to permit persons to whom the Software is | |
34 | ## furnished to do so, subject to the following conditions: | |
35 | ## | |
36 | ## The above copyright notice and this permission notice shall be included in | |
37 | ## all copies or substantial portions of the Software. | |
38 | ## | |
39 | ## THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | |
40 | ## IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
41 | ## FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE | |
42 | ## AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | |
43 | ## LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING | |
44 | ## FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS | |
45 | ## IN THE SOFTWARE. | |
46 | ###################################################################### | |
47 | ||
48 | BOOL_CONST_TYPES = (ast.Constant, ast.NameConstant) | |
49 | AST_CONST_TYPES = (ast.Constant, ast.NameConstant, ast.Str, ast.Num) | |
50 | STR_TYPES = (ast.Constant, ast.Str) | |
51 | ||
52 | ||
53 | class SimplifyNodeVisitor(ast.NodeVisitor): | |
54 | """ | |
55 | Class to traverse the AST node tree and check for code that can be | |
56 | simplified. | |
57 | """ | |
58 | def __init__(self, errorCallback): | |
59 | """ | |
60 | Constructor | |
61 | ||
62 | @param errorCallback callback function to register an error | |
63 | @type func | |
64 | """ | |
65 | super(SimplifyNodeVisitor, self).__init__() | |
66 | ||
67 | self.__error = errorCallback | |
68 | ||
69 | def visit_Expr(self, node): | |
70 | """ | |
71 | Public method to process an Expr node. | |
72 | ||
73 | @param node reference to the Expr node | |
74 | @type ast.Expr | |
75 | """ | |
76 | self.__check112(node) | |
77 | ||
78 | self.generic_visit(node) | |
79 | ||
80 | def visit_BoolOp(self, node): | |
81 | """ | |
82 | Public method to process a BoolOp node. | |
83 | ||
84 | @param node reference to the BoolOp node | |
85 | @type ast.BoolOp | |
86 | """ | |
87 | self.__check101(node) | |
88 | self.__check109(node) | |
89 | ||
90 | self.generic_visit(node) | |
91 | ||
92 | def visit_If(self, node): | |
93 | """ | |
94 | Public method to process an If node. | |
95 | ||
96 | @param node reference to the If node | |
97 | @type ast.If | |
98 | """ | |
99 | self.__check102(node) | |
100 | self.__check103(node) | |
101 | self.__check106(node) | |
102 | self.__check108(node) | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
103 | self.__check114(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
104 | self.__check116(node) |
8189 | 105 | |
106 | self.generic_visit(node) | |
107 | ||
108 | def visit_For(self, node): | |
109 | """ | |
110 | Public method to process a For node. | |
111 | ||
112 | @param node reference to the For node | |
113 | @type ast.For | |
114 | """ | |
115 | self.__check104(node) | |
116 | self.__check110_111(node) | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
117 | self.__check113(node) |
8189 | 118 | |
119 | self.generic_visit(node) | |
120 | ||
121 | def visit_Try(self, node): | |
122 | """ | |
123 | Public method to process a Try node. | |
124 | ||
125 | @param node reference to the Try node | |
126 | @type ast.Try | |
127 | """ | |
128 | self.__check105(node) | |
129 | self.__check107(node) | |
130 | ||
131 | self.generic_visit(node) | |
132 | ||
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
133 | def visit_Call(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
134 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
135 | Public method to process a Call node. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
136 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
137 | @param node reference to the Call node |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
138 | @type ast.Call |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
139 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
140 | self.__check115(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
141 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
142 | self.generic_visit(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
143 | |
8189 | 144 | ############################################################# |
145 | ## Helper methods for the various checkers below | |
146 | ############################################################# | |
147 | ||
148 | def __getDuplicatedIsinstanceCall(self, node): | |
149 | """ | |
150 | Private method to get a list of isinstance arguments which could | |
151 | be combined. | |
152 | ||
153 | @param node reference to the AST node to be inspected | |
154 | @type ast.BoolOp | |
155 | @return list of variable names of duplicated isinstance calls | |
156 | @rtype list of str | |
157 | """ | |
158 | counter = collections.defaultdict(int) | |
159 | ||
160 | for call in node.values: | |
161 | # Ensure this is a call of the built-in isinstance() function. | |
162 | if not isinstance(call, ast.Call) or len(call.args) != 2: | |
163 | continue | |
164 | functionName = unparse(call.func) | |
165 | if functionName != "isinstance": | |
166 | continue | |
167 | ||
168 | arg0Name = unparse(call.args[0]) | |
169 | counter[arg0Name] += 1 | |
170 | ||
171 | return [name for name, count in counter.items() if count > 1] | |
172 | ||
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
173 | def __isConstantIncrease(self, expression): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
174 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
175 | Private method check the given expression for being a constant |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
176 | increase. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
177 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
178 | @param expression reference to the expression node |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
179 | @type ast.AugAssign |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
180 | @return flag indicating a constant increase |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
181 | @rtype bool |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
182 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
183 | return ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
184 | isinstance(expression.op, ast.Add) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
185 | isinstance(expression.value, (ast.Constant, ast.Num)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
186 | ) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
187 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
188 | def __getIfBodyPairs(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
189 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
190 | Private method to extract a list of pairs of test and body for an |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
191 | If node. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
192 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
193 | @param node reference to the If node to be processed |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
194 | @type ast.If |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
195 | @return list of pairs of test and body |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
196 | @rtype list of tuples of (ast.expr, [ast.stmt]) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
197 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
198 | pairs = [(node.test, node.body)] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
199 | orelse = node.orelse |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
200 | while ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
201 | isinstance(orelse, list) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
202 | len(orelse) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
203 | isinstance(orelse[0], ast.If) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
204 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
205 | pairs.append((orelse[0].test, orelse[0].body)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
206 | orelse = orelse[0].orelse |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
207 | return pairs |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
208 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
209 | def __isSameBody(self, body1, body2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
210 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
211 | Private method check, if the given bodies are equivalent. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
212 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
213 | @param body1 list of statements of the first body |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
214 | @type list of ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
215 | @param body2 list of statements of the second body |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
216 | @type list of ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
217 | @return flag indicating identical bodies |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
218 | @rtype bool |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
219 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
220 | if len(body1) != len(body2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
221 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
222 | for a, b in zip(body1, body2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
223 | try: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
224 | statementEqual = self.__isStatementEqual(a, b) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
225 | except RecursionError: # maximum recursion depth |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
226 | statementEqual = False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
227 | if not statementEqual: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
228 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
229 | return True |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
230 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
231 | def __isStatementEqual(self, a: ast.stmt, b: ast.stmt) -> bool: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
232 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
233 | Private method to check, if two statements are equal. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
234 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
235 | @param a reference to the first statement |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
236 | @type ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
237 | @param b reference to the second statement |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
238 | @type ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
239 | @return flag indicating if the two statements are equal |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
240 | @rtype bool |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
241 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
242 | if type(a) is not type(b): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
243 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
244 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
245 | if isinstance(a, ast.AST): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
246 | for k, v in vars(a).items(): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
247 | if k in ("lineno", "col_offset", "ctx", "end_lineno", |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
248 | "parent"): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
249 | continue |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
250 | if not self.__isStatementEqual(v, getattr(b, k)): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
251 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
252 | return True |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
253 | elif isinstance(a, list): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
254 | return all(itertools.starmap(self.__isStatementEqual, zip(a, b))) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
255 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
256 | return a == b |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
257 | |
8189 | 258 | ############################################################# |
259 | ## Methods to check for possible code simplifications below | |
260 | ############################################################# | |
261 | ||
262 | def __check101(self, node): | |
263 | """ | |
264 | Private method to check for duplicate isinstance() calls. | |
265 | ||
266 | @param node reference to the AST node to be checked | |
267 | @type ast.BoolOp | |
268 | """ | |
269 | if isinstance(node.op, ast.Or): | |
270 | for variable in self.__getDuplicatedIsinstanceCall(node): | |
271 | self.__error(node.lineno - 1, node.col_offset, "Y101", | |
272 | variable) | |
273 | ||
274 | def __check102(self, node): | |
275 | """ | |
276 | Private method to check for nested if statements without else blocks. | |
277 | ||
278 | @param node reference to the AST node to be checked | |
279 | @type ast.If | |
280 | """ | |
281 | # ## Pattern 1 | |
282 | # if a: <--- | |
283 | # if b: <--- | |
284 | # c | |
285 | isPattern1 = ( | |
286 | node.orelse == [] and | |
287 | len(node.body) == 1 and | |
288 | isinstance(node.body[0], ast.If) and | |
289 | node.body[0].orelse == [] | |
290 | ) | |
291 | # ## Pattern 2 | |
292 | # if a: < irrelvant for here | |
293 | # pass | |
294 | # elif b: <--- this is treated like a nested block | |
295 | # if c: <--- | |
296 | # d | |
297 | if isPattern1: | |
298 | self.__error(node.lineno - 1, node.col_offset, "Y102") | |
299 | ||
300 | def __check103(self, node): | |
301 | """ | |
302 | Private method to check for calls that wrap a condition to return | |
303 | a bool. | |
304 | ||
305 | @param node reference to the AST node to be checked | |
306 | @type ast.If | |
307 | """ | |
308 | # if cond: | |
309 | # return True | |
310 | # else: | |
311 | # return False | |
312 | if not ( | |
313 | len(node.body) != 1 or | |
314 | not isinstance(node.body[0], ast.Return) or | |
315 | not isinstance(node.body[0].value, BOOL_CONST_TYPES) or | |
316 | not ( | |
317 | node.body[0].value.value is True or | |
318 | node.body[0].value.value is False | |
319 | ) or | |
320 | len(node.orelse) != 1 or | |
321 | not isinstance(node.orelse[0], ast.Return) or | |
322 | not isinstance(node.orelse[0].value, BOOL_CONST_TYPES) or | |
323 | not ( | |
324 | node.orelse[0].value.value is True or | |
325 | node.orelse[0].value.value is False | |
326 | ) | |
327 | ): | |
328 | condition = unparse(node.test) | |
329 | self.__error(node.lineno - 1, node.col_offset, "Y103", condition) | |
330 | ||
331 | def __check104(self, node): | |
332 | """ | |
333 | Private method to check for "iterate and yield" patterns. | |
334 | ||
335 | @param node reference to the AST node to be checked | |
336 | @type ast.For | |
337 | """ | |
338 | # for item in iterable: | |
339 | # yield item | |
340 | if not ( | |
341 | len(node.body) != 1 or | |
342 | not isinstance(node.body[0], ast.Expr) or | |
343 | not isinstance(node.body[0].value, ast.Yield) or | |
344 | not isinstance(node.target, ast.Name) or | |
345 | not isinstance(node.body[0].value.value, ast.Name) or | |
346 | node.target.id != node.body[0].value.value.id or | |
347 | node.orelse != [] | |
348 | ): | |
349 | iterable = unparse(node.iter) | |
350 | self.__error(node.lineno - 1, node.col_offset, "Y104", iterable) | |
351 | ||
352 | def __check105(self, node): | |
353 | """ | |
354 | Private method to check for "try-except-pass" patterns. | |
355 | ||
356 | @param node reference to the AST node to be checked | |
357 | @type ast.Try | |
358 | """ | |
359 | # try: | |
360 | # foo() | |
361 | # except ValueError: | |
362 | # pass | |
363 | if not ( | |
364 | len(node.body) != 1 or | |
365 | len(node.handlers) != 1 or | |
366 | not isinstance(node.handlers[0], ast.ExceptHandler) or | |
367 | len(node.handlers[0].body) != 1 or | |
368 | not isinstance(node.handlers[0].body[0], ast.Pass) or | |
369 | node.orelse != [] | |
370 | ): | |
371 | if node.handlers[0].type is None: | |
372 | exception = "Exception" | |
373 | else: | |
374 | exception = unparse(node.handlers[0].type) | |
375 | self.__error(node.lineno - 1, node.col_offset, "Y105", exception) | |
376 | ||
377 | def __check106(self, node): | |
378 | """ | |
379 | Private method to check for calls where an exception is raised in else. | |
380 | ||
381 | @param node reference to the AST node to be checked | |
382 | @type ast.If | |
383 | """ | |
384 | # if cond: | |
385 | # return True | |
386 | # else: | |
387 | # raise Exception | |
388 | just_one = ( | |
389 | len(node.body) == 1 and | |
390 | len(node.orelse) >= 1 and | |
391 | isinstance(node.orelse[-1], ast.Raise) and | |
392 | not isinstance(node.body[-1], ast.Raise) | |
393 | ) | |
394 | many = ( | |
395 | len(node.body) > 2 * len(node.orelse) and | |
396 | len(node.orelse) >= 1 and | |
397 | isinstance(node.orelse[-1], ast.Raise) and | |
398 | not isinstance(node.body[-1], ast.Raise) | |
399 | ) | |
400 | if just_one or many: | |
401 | self.__error(node.lineno - 1, node.col_offset, "Y106") | |
402 | ||
403 | def __check107(self, node): | |
404 | """ | |
405 | Private method to check for calls where try/except and finally have | |
406 | 'return'. | |
407 | ||
408 | @param node reference to the AST node to be checked | |
409 | @type ast.Try | |
410 | """ | |
411 | # def foo(): | |
412 | # try: | |
413 | # 1 / 0 | |
414 | # return "1" | |
415 | # except: | |
416 | # return "2" | |
417 | # finally: | |
418 | # return "3" | |
419 | tryHasReturn = False | |
420 | for stmt in node.body: | |
421 | if isinstance(stmt, ast.Return): | |
422 | tryHasReturn = True | |
423 | break | |
424 | ||
425 | exceptHasReturn = False | |
426 | for stmt2 in node.handlers: | |
427 | if isinstance(stmt2, ast.Return): | |
428 | exceptHasReturn = True | |
429 | break | |
430 | ||
431 | finallyHasReturn = False | |
432 | finallyReturn = None | |
433 | for stmt in node.finalbody: | |
434 | if isinstance(stmt, ast.Return): | |
435 | finallyHasReturn = True | |
436 | finallyReturn = stmt | |
437 | break | |
438 | ||
439 | if (tryHasReturn or exceptHasReturn) and finallyHasReturn: | |
440 | if finallyReturn is not None: | |
441 | self.__error(finallyReturn.lineno - 1, | |
442 | finallyReturn.col_offset, "Y107") | |
443 | ||
444 | def __check108(self, node): | |
445 | """ | |
446 | Private method to check for if-elses which could be a ternary | |
447 | operator assignment. | |
448 | ||
449 | @param node reference to the AST node to be checked | |
450 | @type ast.If | |
451 | """ | |
452 | # if a: | |
453 | # b = c | |
454 | # else: | |
455 | # b = d | |
456 | if ( | |
457 | len(node.body) == 1 and | |
458 | len(node.orelse) == 1 and | |
459 | isinstance(node.body[0], ast.Assign) and | |
460 | isinstance(node.orelse[0], ast.Assign) and | |
461 | len(node.body[0].targets) == 1 and | |
462 | len(node.orelse[0].targets) == 1 and | |
463 | isinstance(node.body[0].targets[0], ast.Name) and | |
464 | isinstance(node.orelse[0].targets[0], ast.Name) and | |
465 | node.body[0].targets[0].id == node.orelse[0].targets[0].id | |
466 | ): | |
467 | assign = unparse(node.body[0].targets[0]) | |
468 | body = unparse(node.body[0].value) | |
469 | cond = unparse(node.test) | |
470 | orelse = unparse(node.orelse[0].value) | |
471 | self.__error(node.lineno - 1, node.col_offset, "Y108", | |
472 | assign, body, cond, orelse) | |
473 | ||
474 | def __check109(self, node): | |
475 | """ | |
476 | Private method to check for multiple equalities with the same value | |
477 | are combined via "or". | |
478 | ||
479 | @param node reference to the AST node to be checked | |
480 | @type ast.BoolOp | |
481 | """ | |
482 | # if a == b or a == c: | |
483 | # d | |
484 | if isinstance(node.op, ast.Or): | |
485 | equalities = [ | |
486 | value | |
487 | for value in node.values | |
488 | if isinstance(value, ast.Compare) and | |
489 | len(value.ops) == 1 and | |
490 | isinstance(value.ops[0], ast.Eq) | |
491 | ] | |
492 | ids = [] # (name, compared_to) | |
493 | for eq in equalities: | |
494 | if isinstance(eq.left, ast.Name): | |
495 | ids.append((eq.left, eq.comparators[0])) | |
496 | if ( | |
497 | len(eq.comparators) == 1 and | |
498 | isinstance(eq.comparators[0], ast.Name) | |
499 | ): | |
500 | ids.append((eq.comparators[0], eq.left)) | |
501 | ||
502 | id2count = {} | |
503 | for identifier, comparedTo in ids: | |
504 | if identifier.id not in id2count: | |
505 | id2count[identifier.id] = [] | |
506 | id2count[identifier.id].append(comparedTo) | |
507 | for value, values in id2count.items(): | |
508 | if len(values) == 1: | |
509 | continue | |
510 | ||
511 | self.__error(node.lineno - 1, node.col_offset, "Y109", | |
512 | value, unparse(ast.List(elts=values)), | |
513 | unparse(node)) | |
514 | ||
515 | def __check110_111(self, node): | |
516 | """ | |
517 | Private method to check if any / all could be used. | |
518 | ||
519 | @param node reference to the AST node to be checked | |
520 | @type ast.For | |
521 | """ | |
522 | # for x in iterable: | |
523 | # if check(x): | |
524 | # return True | |
525 | # return False | |
526 | # | |
527 | # for x in iterable: | |
528 | # if check(x): | |
529 | # return False | |
530 | # return True | |
531 | if ( | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
532 | len(node.body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
533 | isinstance(node.body[0], ast.If) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
534 | len(node.body[0].body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
535 | isinstance(node.body[0].body[0], ast.Return) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
536 | isinstance(node.body[0].body[0].value, BOOL_CONST_TYPES) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
537 | hasattr(node.body[0].body[0].value, "value") |
8189 | 538 | ): |
539 | check = unparse(node.body[0].test) | |
540 | target = unparse(node.target) | |
541 | iterable = unparse(node.iter) | |
542 | if node.body[0].body[0].value.value is True: | |
543 | self.__error(node.lineno - 1, node.col_offset, "Y110", | |
544 | check, target, iterable) | |
545 | elif node.body[0].body[0].value.value is False: | |
546 | check = "not " + check | |
547 | if check.startswith("not not"): | |
548 | check = check[len("not not "):] | |
549 | self.__error(node.lineno - 1, node.col_offset, "Y111", | |
550 | check, target, iterable) | |
551 | ||
552 | def __check112(self, node): | |
553 | """ | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
554 | Private method to check for non-capitalized calls to environment |
8189 | 555 | variables. |
556 | ||
557 | @param node reference to the AST node to be checked | |
558 | @type ast.Expr | |
559 | """ | |
560 | # os.environ["foo"] | |
561 | # os.environ.get("bar") | |
562 | isIndexCall = ( | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
563 | isinstance(node.value, ast.Subscript) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
564 | isinstance(node.value.value, ast.Attribute) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
565 | isinstance(node.value.value.value, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
566 | node.value.value.value.id == "os" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
567 | node.value.value.attr == "environ" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
568 | ( |
8189 | 569 | ( |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
570 | isinstance(node.value.slice, ast.Index) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
571 | isinstance(node.value.slice.value, STR_TYPES) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
572 | ) or |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
573 | isinstance(node.value.slice, ast.Constant) |
8189 | 574 | ) |
575 | ) | |
576 | if isIndexCall: | |
577 | subscript = node.value | |
578 | slice_ = subscript.slice | |
579 | if isinstance(slice_, ast.Index): | |
580 | # Python < 3.9 | |
581 | stringPart = slice_.value # type: ignore | |
582 | if isinstance(stringPart, ast.Str): | |
583 | envName = stringPart.s # Python 3.6 / 3.7 fallback | |
584 | else: | |
585 | envName = stringPart.value | |
586 | elif isinstance(slice_, ast.Constant): | |
587 | # Python 3.9 | |
588 | envName = slice_.value | |
589 | ||
590 | # Check if this has a change | |
591 | hasChange = envName != envName.upper() | |
592 | ||
593 | isGetCall = ( | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
594 | isinstance(node.value, ast.Call) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
595 | isinstance(node.value.func, ast.Attribute) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
596 | isinstance(node.value.func.value, ast.Attribute) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
597 | isinstance(node.value.func.value.value, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
598 | node.value.func.value.value.id == "os" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
599 | node.value.func.value.attr == "environ" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
600 | node.value.func.attr == "get" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
601 | len(node.value.args) in [1, 2] and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
602 | isinstance(node.value.args[0], STR_TYPES) |
8189 | 603 | ) |
604 | if isGetCall: | |
605 | call = node.value | |
606 | stringPart = call.args[0] | |
607 | if isinstance(stringPart, ast.Str): | |
608 | envName = stringPart.s # Python 3.6 / 3.7 fallback | |
609 | else: | |
610 | envName = stringPart.value | |
611 | # Check if this has a change | |
612 | hasChange = envName != envName.upper() | |
613 | if not (isIndexCall or isGetCall) or not hasChange: | |
614 | return | |
615 | if isIndexCall: | |
616 | original = unparse(node) | |
617 | expected = f"os.environ['{envName.upper()}']" | |
618 | elif isGetCall: | |
619 | original = unparse(node) | |
620 | if len(node.value.args) == 1: | |
621 | expected = f"os.environ.get('{envName.upper()}')" | |
622 | else: | |
623 | defaultValue = unparse(node.value.args[1]) | |
624 | expected = ( | |
625 | f"os.environ.get('{envName.upper()}', '{defaultValue}')" | |
626 | ) | |
627 | else: | |
628 | return | |
629 | ||
630 | self.__error(node.lineno - 1, node.col_offset, "Y112", expected, | |
631 | original) | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
632 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
633 | def __check113(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
634 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
635 | Private method to check for loops in which "enumerate" should be |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
636 | used. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
637 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
638 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
639 | @type ast.For |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
640 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
641 | # idx = 0 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
642 | # for el in iterable: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
643 | # ... |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
644 | # idx += 1 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
645 | variableCandidates = [] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
646 | for expression in node.body: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
647 | if ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
648 | isinstance(expression, ast.AugAssign) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
649 | self.__isConstantIncrease(expression) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
650 | isinstance(expression.target, ast.Name) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
651 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
652 | variableCandidates.append(expression.target) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
653 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
654 | for candidate in variableCandidates: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
655 | self.__error(candidate.lineno - 1, candidate.col_offset, "Y113", |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
656 | unparse(candidate)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
657 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
658 | def __check114(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
659 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
660 | Private method to check for alternative if clauses with identical |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
661 | bodies. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
662 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
663 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
664 | @type ast.If |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
665 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
666 | # if a: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
667 | # b |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
668 | # elif c: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
669 | # b |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
670 | ifBodyPairs = self.__getIfBodyPairs(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
671 | errorPairs = [] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
672 | for ifbody1, ifbody2 in itertools.combinations(ifBodyPairs, 2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
673 | if self.__isSameBody(ifbody1[1], ifbody2[1]): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
674 | errorPairs.append((ifbody1, ifbody2)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
675 | for ifbody1, ifbody2 in errorPairs: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
676 | self.__error(ifbody1[0].lineno - 1, ifbody1[0].col_offset, "Y114", |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
677 | unparse(ifbody1[0]), unparse(ifbody2[0])) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
678 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
679 | def __check115(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
680 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
681 | Private method to to check for places where open() is called without |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
682 | a context handler. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
683 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
684 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
685 | @type ast.Call |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
686 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
687 | # f = open(...) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
688 | #. .. # (do something with f) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
689 | # f.close() |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
690 | if ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
691 | isinstance(node.func, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
692 | node.func.id == "open" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
693 | not isinstance(node.parent, ast.withitem) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
694 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
695 | self.__error(node.lineno - 1, node.col_offset, "Y115") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
696 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
697 | def __check116(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
698 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
699 | Private method to check for places with 3 or more consecutive |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
700 | if-statements with direct returns. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
701 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
702 | * Each if-statement must be a check for equality with the |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
703 | same variable |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
704 | * Each if-statement must just have a "return" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
705 | * Else must also just have a return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
706 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
707 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
708 | @type ast.If |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
709 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
710 | # if a == "foo": |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
711 | # return "bar" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
712 | # elif a == "bar": |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
713 | # return "baz" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
714 | # elif a == "boo": |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
715 | # return "ooh" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
716 | # else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
717 | # return 42 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
718 | if ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
719 | isinstance(node.test, ast.Compare) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
720 | isinstance(node.test.left, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
721 | len(node.test.ops) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
722 | isinstance(node.test.ops[0], ast.Eq) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
723 | len(node.test.comparators) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
724 | isinstance(node.test.comparators[0], AST_CONST_TYPES) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
725 | len(node.body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
726 | isinstance(node.body[0], ast.Return) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
727 | len(node.orelse) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
728 | isinstance(node.orelse[0], ast.If) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
729 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
730 | variable = node.test.left |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
731 | child = node.orelse[0] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
732 | elseValue = None |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
733 | if isinstance(node.test.comparators[0], ast.Str): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
734 | keyValuePairs = { |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
735 | node.test.comparators[0].s: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
736 | unparse(node.body[0].value).strip("'") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
737 | } |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
738 | elif isinstance(node.test.comparators[0], ast.Num): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
739 | keyValuePairs = { |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
740 | node.test.comparators[0].n: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
741 | unparse(node.body[0].value).strip("'") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
742 | } |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
743 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
744 | keyValuePairs = { |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
745 | node.test.comparators[0].value: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
746 | unparse(node.body[0].value).strip("'") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
747 | } |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
748 | while child: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
749 | if not ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
750 | isinstance(child.test, ast.Compare) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
751 | isinstance(child.test.left, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
752 | child.test.left.id == variable.id and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
753 | len(child.test.ops) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
754 | isinstance(child.test.ops[0], ast.Eq) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
755 | len(child.test.comparators) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
756 | isinstance(child.test.comparators[0], AST_CONST_TYPES) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
757 | len(child.body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
758 | isinstance(child.body[0], ast.Return) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
759 | len(child.orelse) <= 1 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
760 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
761 | return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
762 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
763 | if isinstance(child.test.comparators[0], ast.Str): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
764 | key = child.test.comparators[0].s |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
765 | elif isinstance(child.test.comparators[0], ast.Num): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
766 | key = child.test.comparators[0].n |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
767 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
768 | key = child.test.comparators[0].value |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
769 | keyValuePairs[key] = unparse(child.body[0].value).strip("'") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
770 | if len(child.orelse) == 1: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
771 | if isinstance(child.orelse[0], ast.If): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
772 | child = child.orelse[0] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
773 | elif isinstance(child.orelse[0], ast.Return): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
774 | elseValue = unparse(child.orelse[0].value) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
775 | child = None |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
776 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
777 | return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
778 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
779 | child = None |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
780 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
781 | if len(keyValuePairs) < 3: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
782 | return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
783 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
784 | if elseValue: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
785 | ret = f"{keyValuePairs}.get({variable.id}, {elseValue})" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
786 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
787 | ret = f"{keyValuePairs}.get({variable.id})" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
788 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
789 | self.__error(node.lineno - 1, node.col_offset, "Y116", ret) |
8189 | 790 | |
791 | # | |
792 | # eflag: noqa = M891 |