Sat, 03 Apr 2021 10:44:07 +0200
Code Style Checker
- completed the first set of 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) |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
118 | self.__check118(node) |
8189 | 119 | |
120 | self.generic_visit(node) | |
121 | ||
122 | def visit_Try(self, node): | |
123 | """ | |
124 | Public method to process a Try node. | |
125 | ||
126 | @param node reference to the Try node | |
127 | @type ast.Try | |
128 | """ | |
129 | self.__check105(node) | |
130 | self.__check107(node) | |
131 | ||
132 | self.generic_visit(node) | |
133 | ||
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
134 | def visit_Call(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
135 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
136 | Public method to process a Call node. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
137 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
138 | @param node reference to the Call node |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
139 | @type ast.Call |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
140 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
141 | self.__check115(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
142 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
143 | self.generic_visit(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
144 | |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
145 | def visit_With(self, node): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
146 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
147 | Public method to process a With node. |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
148 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
149 | @param node reference to the With node |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
150 | @type ast.With |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
151 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
152 | self.__check117(node) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
153 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
154 | self.generic_visit(node) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
155 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
156 | def visit_Compare(self, node): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
157 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
158 | Public method to process a Compare node. |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
159 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
160 | @param node reference to the Compare node |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
161 | @type ast.Compare |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
162 | """ |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
163 | self.__check118(node) |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
164 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
165 | self.generic_visit(node) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
166 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
167 | def visit_ClassDef(self, node): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
168 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
169 | Public method to process a ClassDef node. |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
170 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
171 | @param node reference to the ClassDef node |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
172 | @type ast.ClassDef |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
173 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
174 | self.__check119(node) |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
175 | self.__check120(node) |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
176 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
177 | self.generic_visit(node) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
178 | |
8189 | 179 | ############################################################# |
180 | ## Helper methods for the various checkers below | |
181 | ############################################################# | |
182 | ||
183 | def __getDuplicatedIsinstanceCall(self, node): | |
184 | """ | |
185 | Private method to get a list of isinstance arguments which could | |
186 | be combined. | |
187 | ||
188 | @param node reference to the AST node to be inspected | |
189 | @type ast.BoolOp | |
190 | @return list of variable names of duplicated isinstance calls | |
191 | @rtype list of str | |
192 | """ | |
193 | counter = collections.defaultdict(int) | |
194 | ||
195 | for call in node.values: | |
196 | # Ensure this is a call of the built-in isinstance() function. | |
197 | if not isinstance(call, ast.Call) or len(call.args) != 2: | |
198 | continue | |
199 | functionName = unparse(call.func) | |
200 | if functionName != "isinstance": | |
201 | continue | |
202 | ||
203 | arg0Name = unparse(call.args[0]) | |
204 | counter[arg0Name] += 1 | |
205 | ||
206 | return [name for name, count in counter.items() if count > 1] | |
207 | ||
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
208 | def __isConstantIncrease(self, expression): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
209 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
210 | Private method check the given expression for being a constant |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
211 | increase. |
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 expression reference to the expression node |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
214 | @type ast.AugAssign |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
215 | @return flag indicating a constant increase |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
216 | @rtype bool |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
217 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
218 | return ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
219 | isinstance(expression.op, ast.Add) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
220 | isinstance(expression.value, (ast.Constant, ast.Num)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
221 | ) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
222 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
223 | def __getIfBodyPairs(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
224 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
225 | 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
|
226 | If node. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
227 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
228 | @param node reference to the If node to be processed |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
229 | @type ast.If |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
230 | @return list of pairs of test and body |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
231 | @rtype list of tuples of (ast.expr, [ast.stmt]) |
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 | pairs = [(node.test, node.body)] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
234 | orelse = node.orelse |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
235 | while ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
236 | isinstance(orelse, list) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
237 | len(orelse) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
238 | isinstance(orelse[0], ast.If) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
239 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
240 | pairs.append((orelse[0].test, orelse[0].body)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
241 | orelse = orelse[0].orelse |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
242 | return pairs |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
243 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
244 | def __isSameBody(self, body1, body2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
245 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
246 | Private method check, if the given bodies are equivalent. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
247 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
248 | @param body1 list of statements of the first body |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
249 | @type list of ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
250 | @param body2 list of statements of the second body |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
251 | @type list of ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
252 | @return flag indicating identical bodies |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
253 | @rtype bool |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
254 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
255 | if len(body1) != len(body2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
256 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
257 | for a, b in zip(body1, body2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
258 | try: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
259 | statementEqual = self.__isStatementEqual(a, b) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
260 | except RecursionError: # maximum recursion depth |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
261 | statementEqual = False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
262 | if not statementEqual: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
263 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
264 | return True |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
265 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
266 | def __isStatementEqual(self, a: ast.stmt, b: ast.stmt) -> bool: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
267 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
268 | Private method to check, if two statements are equal. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
269 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
270 | @param a reference to the first statement |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
271 | @type ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
272 | @param b reference to the second statement |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
273 | @type ast.stmt |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
274 | @return flag indicating if the two statements are equal |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
275 | @rtype bool |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
276 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
277 | if type(a) is not type(b): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
278 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
279 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
280 | if isinstance(a, ast.AST): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
281 | for k, v in vars(a).items(): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
282 | if k in ("lineno", "col_offset", "ctx", "end_lineno", |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
283 | "parent"): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
284 | continue |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
285 | if not self.__isStatementEqual(v, getattr(b, k)): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
286 | return False |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
287 | return True |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
288 | elif isinstance(a, list): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
289 | return all(itertools.starmap(self.__isStatementEqual, zip(a, b))) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
290 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
291 | return a == b |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
292 | |
8189 | 293 | ############################################################# |
294 | ## Methods to check for possible code simplifications below | |
295 | ############################################################# | |
296 | ||
297 | def __check101(self, node): | |
298 | """ | |
299 | Private method to check for duplicate isinstance() calls. | |
300 | ||
301 | @param node reference to the AST node to be checked | |
302 | @type ast.BoolOp | |
303 | """ | |
304 | if isinstance(node.op, ast.Or): | |
305 | for variable in self.__getDuplicatedIsinstanceCall(node): | |
306 | self.__error(node.lineno - 1, node.col_offset, "Y101", | |
307 | variable) | |
308 | ||
309 | def __check102(self, node): | |
310 | """ | |
311 | Private method to check for nested if statements without else blocks. | |
312 | ||
313 | @param node reference to the AST node to be checked | |
314 | @type ast.If | |
315 | """ | |
316 | # ## Pattern 1 | |
317 | # if a: <--- | |
318 | # if b: <--- | |
319 | # c | |
320 | isPattern1 = ( | |
321 | node.orelse == [] and | |
322 | len(node.body) == 1 and | |
323 | isinstance(node.body[0], ast.If) and | |
324 | node.body[0].orelse == [] | |
325 | ) | |
326 | # ## Pattern 2 | |
327 | # if a: < irrelvant for here | |
328 | # pass | |
329 | # elif b: <--- this is treated like a nested block | |
330 | # if c: <--- | |
331 | # d | |
332 | if isPattern1: | |
333 | self.__error(node.lineno - 1, node.col_offset, "Y102") | |
334 | ||
335 | def __check103(self, node): | |
336 | """ | |
337 | Private method to check for calls that wrap a condition to return | |
338 | a bool. | |
339 | ||
340 | @param node reference to the AST node to be checked | |
341 | @type ast.If | |
342 | """ | |
343 | # if cond: | |
344 | # return True | |
345 | # else: | |
346 | # return False | |
347 | if not ( | |
348 | len(node.body) != 1 or | |
349 | not isinstance(node.body[0], ast.Return) or | |
350 | not isinstance(node.body[0].value, BOOL_CONST_TYPES) or | |
351 | not ( | |
352 | node.body[0].value.value is True or | |
353 | node.body[0].value.value is False | |
354 | ) or | |
355 | len(node.orelse) != 1 or | |
356 | not isinstance(node.orelse[0], ast.Return) or | |
357 | not isinstance(node.orelse[0].value, BOOL_CONST_TYPES) or | |
358 | not ( | |
359 | node.orelse[0].value.value is True or | |
360 | node.orelse[0].value.value is False | |
361 | ) | |
362 | ): | |
363 | condition = unparse(node.test) | |
364 | self.__error(node.lineno - 1, node.col_offset, "Y103", condition) | |
365 | ||
366 | def __check104(self, node): | |
367 | """ | |
368 | Private method to check for "iterate and yield" patterns. | |
369 | ||
370 | @param node reference to the AST node to be checked | |
371 | @type ast.For | |
372 | """ | |
373 | # for item in iterable: | |
374 | # yield item | |
375 | if not ( | |
376 | len(node.body) != 1 or | |
377 | not isinstance(node.body[0], ast.Expr) or | |
378 | not isinstance(node.body[0].value, ast.Yield) or | |
379 | not isinstance(node.target, ast.Name) or | |
380 | not isinstance(node.body[0].value.value, ast.Name) or | |
381 | node.target.id != node.body[0].value.value.id or | |
382 | node.orelse != [] | |
383 | ): | |
384 | iterable = unparse(node.iter) | |
385 | self.__error(node.lineno - 1, node.col_offset, "Y104", iterable) | |
386 | ||
387 | def __check105(self, node): | |
388 | """ | |
389 | Private method to check for "try-except-pass" patterns. | |
390 | ||
391 | @param node reference to the AST node to be checked | |
392 | @type ast.Try | |
393 | """ | |
394 | # try: | |
395 | # foo() | |
396 | # except ValueError: | |
397 | # pass | |
398 | if not ( | |
399 | len(node.body) != 1 or | |
400 | len(node.handlers) != 1 or | |
401 | not isinstance(node.handlers[0], ast.ExceptHandler) or | |
402 | len(node.handlers[0].body) != 1 or | |
403 | not isinstance(node.handlers[0].body[0], ast.Pass) or | |
404 | node.orelse != [] | |
405 | ): | |
406 | if node.handlers[0].type is None: | |
407 | exception = "Exception" | |
408 | else: | |
409 | exception = unparse(node.handlers[0].type) | |
410 | self.__error(node.lineno - 1, node.col_offset, "Y105", exception) | |
411 | ||
412 | def __check106(self, node): | |
413 | """ | |
414 | Private method to check for calls where an exception is raised in else. | |
415 | ||
416 | @param node reference to the AST node to be checked | |
417 | @type ast.If | |
418 | """ | |
419 | # if cond: | |
420 | # return True | |
421 | # else: | |
422 | # raise Exception | |
423 | just_one = ( | |
424 | len(node.body) == 1 and | |
425 | len(node.orelse) >= 1 and | |
426 | isinstance(node.orelse[-1], ast.Raise) and | |
427 | not isinstance(node.body[-1], ast.Raise) | |
428 | ) | |
429 | many = ( | |
430 | len(node.body) > 2 * len(node.orelse) and | |
431 | len(node.orelse) >= 1 and | |
432 | isinstance(node.orelse[-1], ast.Raise) and | |
433 | not isinstance(node.body[-1], ast.Raise) | |
434 | ) | |
435 | if just_one or many: | |
436 | self.__error(node.lineno - 1, node.col_offset, "Y106") | |
437 | ||
438 | def __check107(self, node): | |
439 | """ | |
440 | Private method to check for calls where try/except and finally have | |
441 | 'return'. | |
442 | ||
443 | @param node reference to the AST node to be checked | |
444 | @type ast.Try | |
445 | """ | |
446 | # def foo(): | |
447 | # try: | |
448 | # 1 / 0 | |
449 | # return "1" | |
450 | # except: | |
451 | # return "2" | |
452 | # finally: | |
453 | # return "3" | |
454 | tryHasReturn = False | |
455 | for stmt in node.body: | |
456 | if isinstance(stmt, ast.Return): | |
457 | tryHasReturn = True | |
458 | break | |
459 | ||
460 | exceptHasReturn = False | |
461 | for stmt2 in node.handlers: | |
462 | if isinstance(stmt2, ast.Return): | |
463 | exceptHasReturn = True | |
464 | break | |
465 | ||
466 | finallyHasReturn = False | |
467 | finallyReturn = None | |
468 | for stmt in node.finalbody: | |
469 | if isinstance(stmt, ast.Return): | |
470 | finallyHasReturn = True | |
471 | finallyReturn = stmt | |
472 | break | |
473 | ||
474 | if (tryHasReturn or exceptHasReturn) and finallyHasReturn: | |
475 | if finallyReturn is not None: | |
476 | self.__error(finallyReturn.lineno - 1, | |
477 | finallyReturn.col_offset, "Y107") | |
478 | ||
479 | def __check108(self, node): | |
480 | """ | |
481 | Private method to check for if-elses which could be a ternary | |
482 | operator assignment. | |
483 | ||
484 | @param node reference to the AST node to be checked | |
485 | @type ast.If | |
486 | """ | |
487 | # if a: | |
488 | # b = c | |
489 | # else: | |
490 | # b = d | |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
491 | # |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
492 | # but not: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
493 | # if a: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
494 | # b = c |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
495 | # elif c: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
496 | # b = e |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
497 | # else: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
498 | # b = d |
8189 | 499 | if ( |
500 | len(node.body) == 1 and | |
501 | len(node.orelse) == 1 and | |
502 | isinstance(node.body[0], ast.Assign) and | |
503 | isinstance(node.orelse[0], ast.Assign) and | |
504 | len(node.body[0].targets) == 1 and | |
505 | len(node.orelse[0].targets) == 1 and | |
506 | isinstance(node.body[0].targets[0], ast.Name) and | |
507 | isinstance(node.orelse[0].targets[0], ast.Name) and | |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
508 | node.body[0].targets[0].id == node.orelse[0].targets[0].id and |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
509 | not isinstance(node.parent, ast.If) |
8189 | 510 | ): |
511 | assign = unparse(node.body[0].targets[0]) | |
512 | body = unparse(node.body[0].value) | |
513 | cond = unparse(node.test) | |
514 | orelse = unparse(node.orelse[0].value) | |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
515 | if len( |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
516 | "{0} = {1} if {2} else {3}".format(assign, body, cond, orelse) |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
517 | ) < 79: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
518 | # don't flag an issue, if the ternary would get too complicated |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
519 | self.__error(node.lineno - 1, node.col_offset, "Y108", |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
520 | assign, body, cond, orelse) |
8189 | 521 | |
522 | def __check109(self, node): | |
523 | """ | |
524 | Private method to check for multiple equalities with the same value | |
525 | are combined via "or". | |
526 | ||
527 | @param node reference to the AST node to be checked | |
528 | @type ast.BoolOp | |
529 | """ | |
530 | # if a == b or a == c: | |
531 | # d | |
532 | if isinstance(node.op, ast.Or): | |
533 | equalities = [ | |
534 | value | |
535 | for value in node.values | |
536 | if isinstance(value, ast.Compare) and | |
537 | len(value.ops) == 1 and | |
538 | isinstance(value.ops[0], ast.Eq) | |
539 | ] | |
540 | ids = [] # (name, compared_to) | |
541 | for eq in equalities: | |
542 | if isinstance(eq.left, ast.Name): | |
543 | ids.append((eq.left, eq.comparators[0])) | |
544 | if ( | |
545 | len(eq.comparators) == 1 and | |
546 | isinstance(eq.comparators[0], ast.Name) | |
547 | ): | |
548 | ids.append((eq.comparators[0], eq.left)) | |
549 | ||
550 | id2count = {} | |
551 | for identifier, comparedTo in ids: | |
552 | if identifier.id not in id2count: | |
553 | id2count[identifier.id] = [] | |
554 | id2count[identifier.id].append(comparedTo) | |
555 | for value, values in id2count.items(): | |
556 | if len(values) == 1: | |
557 | continue | |
558 | ||
559 | self.__error(node.lineno - 1, node.col_offset, "Y109", | |
560 | value, unparse(ast.List(elts=values)), | |
561 | unparse(node)) | |
562 | ||
563 | def __check110_111(self, node): | |
564 | """ | |
565 | Private method to check if any / all could be used. | |
566 | ||
567 | @param node reference to the AST node to be checked | |
568 | @type ast.For | |
569 | """ | |
570 | # for x in iterable: | |
571 | # if check(x): | |
572 | # return True | |
573 | # return False | |
574 | # | |
575 | # for x in iterable: | |
576 | # if check(x): | |
577 | # return False | |
578 | # return True | |
579 | if ( | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
580 | len(node.body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
581 | isinstance(node.body[0], ast.If) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
582 | len(node.body[0].body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
583 | isinstance(node.body[0].body[0], ast.Return) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
584 | 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
|
585 | hasattr(node.body[0].body[0].value, "value") |
8189 | 586 | ): |
587 | check = unparse(node.body[0].test) | |
588 | target = unparse(node.target) | |
589 | iterable = unparse(node.iter) | |
590 | if node.body[0].body[0].value.value is True: | |
591 | self.__error(node.lineno - 1, node.col_offset, "Y110", | |
592 | check, target, iterable) | |
593 | elif node.body[0].body[0].value.value is False: | |
594 | check = "not " + check | |
595 | if check.startswith("not not"): | |
596 | check = check[len("not not "):] | |
597 | self.__error(node.lineno - 1, node.col_offset, "Y111", | |
598 | check, target, iterable) | |
599 | ||
600 | def __check112(self, node): | |
601 | """ | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
602 | Private method to check for non-capitalized calls to environment |
8189 | 603 | variables. |
604 | ||
605 | @param node reference to the AST node to be checked | |
606 | @type ast.Expr | |
607 | """ | |
608 | # os.environ["foo"] | |
609 | # os.environ.get("bar") | |
610 | isIndexCall = ( | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
611 | isinstance(node.value, ast.Subscript) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
612 | isinstance(node.value.value, ast.Attribute) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
613 | isinstance(node.value.value.value, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
614 | node.value.value.value.id == "os" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
615 | node.value.value.attr == "environ" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
616 | ( |
8189 | 617 | ( |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
618 | isinstance(node.value.slice, ast.Index) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
619 | isinstance(node.value.slice.value, STR_TYPES) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
620 | ) or |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
621 | isinstance(node.value.slice, ast.Constant) |
8189 | 622 | ) |
623 | ) | |
624 | if isIndexCall: | |
625 | subscript = node.value | |
626 | slice_ = subscript.slice | |
627 | if isinstance(slice_, ast.Index): | |
628 | # Python < 3.9 | |
629 | stringPart = slice_.value # type: ignore | |
630 | if isinstance(stringPart, ast.Str): | |
631 | envName = stringPart.s # Python 3.6 / 3.7 fallback | |
632 | else: | |
633 | envName = stringPart.value | |
634 | elif isinstance(slice_, ast.Constant): | |
635 | # Python 3.9 | |
636 | envName = slice_.value | |
637 | ||
638 | # Check if this has a change | |
639 | hasChange = envName != envName.upper() | |
640 | ||
641 | isGetCall = ( | |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
642 | isinstance(node.value, ast.Call) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
643 | isinstance(node.value.func, ast.Attribute) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
644 | isinstance(node.value.func.value, ast.Attribute) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
645 | isinstance(node.value.func.value.value, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
646 | node.value.func.value.value.id == "os" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
647 | node.value.func.value.attr == "environ" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
648 | node.value.func.attr == "get" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
649 | len(node.value.args) in [1, 2] and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
650 | isinstance(node.value.args[0], STR_TYPES) |
8189 | 651 | ) |
652 | if isGetCall: | |
653 | call = node.value | |
654 | stringPart = call.args[0] | |
655 | if isinstance(stringPart, ast.Str): | |
656 | envName = stringPart.s # Python 3.6 / 3.7 fallback | |
657 | else: | |
658 | envName = stringPart.value | |
659 | # Check if this has a change | |
660 | hasChange = envName != envName.upper() | |
661 | if not (isIndexCall or isGetCall) or not hasChange: | |
662 | return | |
663 | if isIndexCall: | |
664 | original = unparse(node) | |
665 | expected = f"os.environ['{envName.upper()}']" | |
666 | elif isGetCall: | |
667 | original = unparse(node) | |
668 | if len(node.value.args) == 1: | |
669 | expected = f"os.environ.get('{envName.upper()}')" | |
670 | else: | |
671 | defaultValue = unparse(node.value.args[1]) | |
672 | expected = ( | |
673 | f"os.environ.get('{envName.upper()}', '{defaultValue}')" | |
674 | ) | |
675 | else: | |
676 | return | |
677 | ||
678 | self.__error(node.lineno - 1, node.col_offset, "Y112", expected, | |
679 | original) | |
8191
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 | def __check113(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
682 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
683 | 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
|
684 | used. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
685 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
686 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
687 | @type ast.For |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
688 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
689 | # idx = 0 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
690 | # for el in iterable: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
691 | # ... |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
692 | # idx += 1 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
693 | variableCandidates = [] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
694 | for expression in node.body: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
695 | if ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
696 | isinstance(expression, ast.AugAssign) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
697 | self.__isConstantIncrease(expression) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
698 | isinstance(expression.target, ast.Name) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
699 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
700 | variableCandidates.append(expression.target) |
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 | for candidate in variableCandidates: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
703 | self.__error(candidate.lineno - 1, candidate.col_offset, "Y113", |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
704 | unparse(candidate)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
705 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
706 | def __check114(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
707 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
708 | Private method to check for alternative if clauses with identical |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
709 | bodies. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
710 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
711 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
712 | @type ast.If |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
713 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
714 | # if a: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
715 | # b |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
716 | # elif c: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
717 | # b |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
718 | ifBodyPairs = self.__getIfBodyPairs(node) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
719 | errorPairs = [] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
720 | for ifbody1, ifbody2 in itertools.combinations(ifBodyPairs, 2): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
721 | if self.__isSameBody(ifbody1[1], ifbody2[1]): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
722 | errorPairs.append((ifbody1, ifbody2)) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
723 | for ifbody1, ifbody2 in errorPairs: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
724 | 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
|
725 | unparse(ifbody1[0]), unparse(ifbody2[0])) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
726 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
727 | def __check115(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
728 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
729 | 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
|
730 | a context handler. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
731 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
732 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
733 | @type ast.Call |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
734 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
735 | # f = open(...) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
736 | #. .. # (do something with f) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
737 | # f.close() |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
738 | if ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
739 | isinstance(node.func, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
740 | node.func.id == "open" and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
741 | not isinstance(node.parent, ast.withitem) |
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 | self.__error(node.lineno - 1, node.col_offset, "Y115") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
744 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
745 | def __check116(self, node): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
746 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
747 | 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
|
748 | if-statements with direct returns. |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
749 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
750 | * 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
|
751 | same variable |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
752 | * Each if-statement must just have a "return" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
753 | * Else must also just have a return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
754 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
755 | @param node reference to the AST node to be checked |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
756 | @type ast.If |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
757 | """ |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
758 | # if a == "foo": |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
759 | # return "bar" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
760 | # elif a == "bar": |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
761 | # return "baz" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
762 | # elif a == "boo": |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
763 | # return "ooh" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
764 | # else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
765 | # return 42 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
766 | if ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
767 | isinstance(node.test, ast.Compare) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
768 | isinstance(node.test.left, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
769 | len(node.test.ops) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
770 | isinstance(node.test.ops[0], ast.Eq) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
771 | len(node.test.comparators) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
772 | isinstance(node.test.comparators[0], AST_CONST_TYPES) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
773 | len(node.body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
774 | isinstance(node.body[0], ast.Return) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
775 | len(node.orelse) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
776 | isinstance(node.orelse[0], ast.If) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
777 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
778 | variable = node.test.left |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
779 | child = node.orelse[0] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
780 | elseValue = None |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
781 | if node.body[0].value is not None: |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
782 | bodyValueStr = unparse(node.body[0].value).strip("'") |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
783 | else: |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
784 | bodyValueStr = "None" |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
785 | if isinstance(node.test.comparators[0], ast.Str): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
786 | keyValuePairs = { |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
787 | node.test.comparators[0].s: bodyValueStr |
8191
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 | elif isinstance(node.test.comparators[0], ast.Num): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
790 | keyValuePairs = { |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
791 | node.test.comparators[0].n: bodyValueStr, |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
792 | } |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
793 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
794 | keyValuePairs = { |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
795 | node.test.comparators[0].value: bodyValueStr |
8191
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
796 | } |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
797 | while child: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
798 | if not ( |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
799 | isinstance(child.test, ast.Compare) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
800 | isinstance(child.test.left, ast.Name) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
801 | child.test.left.id == variable.id and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
802 | len(child.test.ops) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
803 | isinstance(child.test.ops[0], ast.Eq) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
804 | len(child.test.comparators) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
805 | isinstance(child.test.comparators[0], AST_CONST_TYPES) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
806 | len(child.body) == 1 and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
807 | isinstance(child.body[0], ast.Return) and |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
808 | len(child.orelse) <= 1 |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
809 | ): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
810 | return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
811 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
812 | if isinstance(child.test.comparators[0], ast.Str): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
813 | key = child.test.comparators[0].s |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
814 | elif isinstance(child.test.comparators[0], ast.Num): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
815 | key = child.test.comparators[0].n |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
816 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
817 | key = child.test.comparators[0].value |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
818 | keyValuePairs[key] = unparse(child.body[0].value).strip("'") |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
819 | if len(child.orelse) == 1: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
820 | if isinstance(child.orelse[0], ast.If): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
821 | child = child.orelse[0] |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
822 | elif isinstance(child.orelse[0], ast.Return): |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
823 | elseValue = unparse(child.orelse[0].value) |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
824 | child = None |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
825 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
826 | return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
827 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
828 | child = None |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
829 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
830 | if len(keyValuePairs) < 3: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
831 | return |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
832 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
833 | if elseValue: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
834 | ret = f"{keyValuePairs}.get({variable.id}, {elseValue})" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
835 | else: |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
836 | ret = f"{keyValuePairs}.get({variable.id})" |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
837 | |
9125da0c227e
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8189
diff
changeset
|
838 | self.__error(node.lineno - 1, node.col_offset, "Y116", ret) |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
839 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
840 | def __check117(self, node): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
841 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
842 | Private method to check for multiple with-statements with same scope. |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
843 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
844 | @param node reference to the AST node to be checked |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
845 | @type ast.With |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
846 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
847 | # with A() as a: |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
848 | # with B() as b: |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
849 | # print("hello") |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
850 | if ( |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
851 | len(node.body) == 1 and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
852 | isinstance(node.body[0], ast.With) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
853 | ): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
854 | withItems = [] |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
855 | for withitem in node.items + node.body[0].items: |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
856 | withItems.append(f"{unparse(withitem)}") |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
857 | mergedWith = f"with {', '.join(withItems)}:" |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
858 | self.__error(node.lineno - 1, node.col_offset, "Y117", mergedWith) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
859 | |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
860 | def __check118(self, node): |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
861 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
862 | Private method to check for usages of "key in dict.keys()". |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
863 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
864 | @param node reference to the AST node to be checked |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
865 | @type ast.Compare or ast.For |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
866 | """ |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
867 | # Pattern 1: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
868 | # |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
869 | # if key in dict.keys(): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
870 | # # do something |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
871 | # |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
872 | # Pattern 2: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
873 | # |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
874 | # for key in dict.keys(): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
875 | # # do something |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
876 | if ( |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
877 | isinstance(node, ast.Compare) and |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
878 | len(node.ops) == 1 and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
879 | isinstance(node.ops[0], ast.In) and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
880 | len(node.comparators) == 1 |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
881 | ): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
882 | callNode = node.comparators[0] |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
883 | elif ( |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
884 | isinstance(node, ast.For) |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
885 | ): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
886 | callNode = node.iter |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
887 | else: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
888 | callNode = None |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
889 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
890 | if not isinstance(callNode, ast.Call): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
891 | return |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
892 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
893 | attrNode = callNode.func |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
894 | if ( |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
895 | isinstance(callNode.func, ast.Attribute) and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
896 | callNode.func.attr == "keys" and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
897 | isinstance(callNode.func.ctx, ast.Load) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
898 | ): |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
899 | if isinstance(node, ast.Compare): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
900 | keyStr = unparse(node.left) |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
901 | else: |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
902 | keyStr = unparse(node.target) |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
903 | dictStr = unparse(attrNode.value) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
904 | self.__error(node.lineno - 1, node.col_offset, "Y118", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
905 | keyStr, dictStr) |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
906 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
907 | def __check119(self, node): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
908 | """ |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
909 | Private method to check for classes that should be "dataclasses". |
8192
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
910 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
911 | @param node reference to the AST node to be checked |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
912 | @type ast.ClassDef |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
913 | """ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
914 | if ( |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
915 | len(node.decorator_list) == 0 and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
916 | len(node.bases) == 0 |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
917 | ): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
918 | dataclassFunctions = [ |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
919 | "__init__", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
920 | "__eq__", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
921 | "__hash__", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
922 | "__repr__", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
923 | "__str__", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
924 | ] |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
925 | hasOnlyConstructorMethod = True |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
926 | for bodyElement in node.body: |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
927 | if ( |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
928 | isinstance(bodyElement, ast.FunctionDef) and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
929 | bodyElement.name not in dataclassFunctions |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
930 | ): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
931 | hasOnlyConstructorMethod = False |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
932 | break |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
933 | |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
934 | if ( |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
935 | hasOnlyConstructorMethod and |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
936 | sum(1 for el in node.body |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
937 | if isinstance(el, ast.FunctionDef)) > 0 |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
938 | ): |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
939 | self.__error(node.lineno - 1, node.col_offset, "Y119", |
e1157bd8b4c2
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8191
diff
changeset
|
940 | node.name) |
8194
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
941 | |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
942 | def __check120(self, node): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
943 | """ |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
944 | Private method to check for classes that inherit from object. |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
945 | |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
946 | @param node reference to the AST node to be checked |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
947 | @type ast.ClassDef |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
948 | """ |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
949 | # class FooBar(object): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
950 | # ... |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
951 | if ( |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
952 | len(node.bases) == 1 and |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
953 | isinstance(node.bases[0], ast.Name) and |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
954 | node.bases[0].id == "object" |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
955 | ): |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
956 | self.__error(node.lineno - 1, node.col_offset, "Y120", |
b925628bf91f
Code Style Checker
Detlev Offenbach <detlev@die-offenbachs.de>
parents:
8192
diff
changeset
|
957 | node.name) |
8189 | 958 | |
959 | # | |
960 | # eflag: noqa = M891 |