| OLD | NEW |
| 1 # This file is part of Adblock Plus <https://adblockplus.org/>, | 1 # This file is part of Adblock Plus <https://adblockplus.org/>, |
| 2 # Copyright (C) 2006-present eyeo GmbH | 2 # Copyright (C) 2006-present eyeo GmbH |
| 3 # | 3 # |
| 4 # Adblock Plus is free software: you can redistribute it and/or modify | 4 # Adblock Plus is free software: you can redistribute it and/or modify |
| 5 # it under the terms of the GNU General Public License version 3 as | 5 # it under the terms of the GNU General Public License version 3 as |
| 6 # published by the Free Software Foundation. | 6 # published by the Free Software Foundation. |
| 7 # | 7 # |
| 8 # Adblock Plus is distributed in the hope that it will be useful, | 8 # Adblock Plus is distributed in the hope that it will be useful, |
| 9 # but WITHOUT ANY WARRANTY; without even the implied warranty of | 9 # but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 10 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| (...skipping 50 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 61 if isinstance(node, ast.Name): | 61 if isinstance(node, ast.Name): |
| 62 return node.id | 62 return node.id |
| 63 if isinstance(node, ast.Attribute) and isinstance(node.value, ast.Name): | 63 if isinstance(node, ast.Attribute) and isinstance(node.value, ast.Name): |
| 64 return '{}.{}'.format(node.value.id, node.attr) | 64 return '{}.{}'.format(node.value.id, node.attr) |
| 65 | 65 |
| 66 | 66 |
| 67 def get_statement(node): | 67 def get_statement(node): |
| 68 return type(node).__name__.lower() | 68 return type(node).__name__.lower() |
| 69 | 69 |
| 70 | 70 |
| 71 def get_descendant_nodes(node): |
| 72 for child in ast.iter_child_nodes(node): |
| 73 yield (child, node) |
| 74 for nodes in get_descendant_nodes(child): |
| 75 yield nodes |
| 76 |
| 77 |
| 71 class TreeVisitor(ast.NodeVisitor): | 78 class TreeVisitor(ast.NodeVisitor): |
| 72 Scope = collections.namedtuple('Scope', ['node', 'names', 'globals']) | 79 Scope = collections.namedtuple('Scope', ['node', 'names', 'globals']) |
| 73 | 80 |
| 74 def __init__(self): | 81 def __init__(self): |
| 75 self.errors = [] | 82 self.errors = [] |
| 76 self.scope_stack = [] | 83 self.scope_stack = [] |
| 77 | 84 |
| 78 def _visit_block(self, nodes, block_required=False, | 85 def _visit_block(self, nodes, block_required=False, |
| 79 nodes_required=True, docstring=False, | 86 nodes_required=True, docstring=False, |
| 80 can_have_unused_expr=False): | 87 can_have_unused_expr=False): |
| (...skipping 330 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 411 prefix = 'u' | 418 prefix = 'u' |
| 412 | 419 |
| 413 literal = '{0}{1}{2}{1}'.format(prefix, quote, text) | 420 literal = '{0}{1}{2}{1}'.format(prefix, quote, text) |
| 414 if ascii(eval(literal)) != literal: | 421 if ascii(eval(literal)) != literal: |
| 415 yield (start, "A110 string literal doesn't match " | 422 yield (start, "A110 string literal doesn't match " |
| 416 '{}()'.format(ascii.__name__)) | 423 '{}()'.format(ascii.__name__)) |
| 417 | 424 |
| 418 first_token = False | 425 first_token = False |
| 419 | 426 |
| 420 | 427 |
| 421 def check_redundant_parenthesis(logical_line, tokens): | 428 def check_redundant_parenthesis(tree, lines, file_tokens): |
| 422 start_line = tokens[0][2][0] | 429 orig = ast.dump(tree) |
| 423 level = 0 | 430 nodes = get_descendant_nodes(tree) |
| 424 statement = None | 431 stack = [] |
| 425 | 432 |
| 426 for i, (kind, token, _, end, _) in enumerate(tokens): | 433 for i, (kind, token, _, _, _) in enumerate(file_tokens): |
| 427 if kind == tokenize.INDENT or kind == tokenize.DEDENT: | 434 if kind != tokenize.OP: |
| 428 continue | 435 continue |
| 429 | 436 |
| 430 if statement is None: | 437 if token == '(': |
| 431 # logical line doesn't start with an if, elif or while statement | 438 stack.append(i) |
| 432 if kind != tokenize.NAME or token not in {'if', 'elif', 'while'}: | 439 elif token == ')': |
| 433 break | 440 start = stack.pop() |
| 441 sample = lines[:] |
| 434 | 442 |
| 435 # expression doesn't start with parenthesis | 443 for pos in [i, start]: |
| 436 next_token = tokens[i + 1] | 444 _, _, (lineno, col1), (_, col2), _ = file_tokens[pos] |
| 437 if next_token[:2] != (tokenize.OP, '('): | 445 lineno -= 1 |
| 438 break | 446 sample[lineno] = (sample[lineno][:col1] + |
| 447 sample[lineno][col2:]) |
| 439 | 448 |
| 440 # expression is empty tuple | 449 try: |
| 441 if tokens[i + 2][:2] == (tokenize.OP, ')'): | 450 modified = ast.parse(''.join(sample)) |
| 442 break | 451 except SyntaxError: |
| 452 # Parentheses syntactically required. |
| 453 continue |
| 443 | 454 |
| 444 statement = token | 455 # Parentheses logically required. |
| 445 pos = next_token[2] | 456 if orig != ast.dump(modified): |
| 446 continue | 457 continue |
| 447 | 458 |
| 448 # expression ends on a different line, parenthesis are necessary | 459 pos = file_tokens[start][2] |
| 449 if end[0] > start_line: | 460 while True: |
| 450 break | 461 node, parent = next(nodes) |
| 462 if pos < (getattr(node, 'lineno', -1), |
| 463 getattr(node, 'col_offset', -1)): |
| 464 break |
| 451 | 465 |
| 452 if kind == tokenize.OP: | 466 # Allow redundant parentheses for readability, |
| 453 if token == ',': | 467 # when creating tuples (but not when unpacking variables), |
| 454 # expression is non-empty tuple | 468 # nested operations and comparisons inside assignments. |
| 455 if level == 1: | 469 is_tuple = ( |
| 456 break | 470 isinstance(node, ast.Tuple) and not ( |
| 457 elif token == '(': | 471 isinstance(parent, (ast.For, ast.comprehension)) and |
| 458 level += 1 | 472 node == parent.target or |
| 459 elif token == ')': | 473 isinstance(parent, ast.Assign) and |
| 460 level -= 1 | 474 node in parent.targets |
| 461 if level == 0: | 475 ) |
| 462 # outer parenthesis closed before end of expression | 476 ) |
| 463 if tokens[i + 1][:2] != (tokenize.OP, ':'): | 477 is_nested_op = ( |
| 464 break | 478 isinstance(node, (ast.BinOp, ast.BoolOp)) and |
| 479 isinstance(parent, (ast.BinOp, ast.BoolOp)) |
| 480 ) |
| 481 is_compare_in_assign = ( |
| 482 isinstance(parent, (ast.Assign, ast.keyword)) and |
| 483 any(isinstance(x, ast.Compare) for x in ast.walk(node)) |
| 484 ) |
| 485 if is_tuple or is_nested_op or is_compare_in_assign: |
| 486 continue |
| 465 | 487 |
| 466 return [(pos, 'A111 redundant parenthesis for {} ' | 488 yield (pos[0], pos[1], 'A111 redundant parenthesis', None) |
| 467 'statement'.format(statement))] | |
| 468 | |
| 469 return [] | |
| 470 | 489 |
| 471 | 490 |
| 472 for checker in [check_ast, check_non_default_encoding, | 491 for checker in [check_ast, check_non_default_encoding, |
| 473 check_quotes, check_redundant_parenthesis]: | 492 check_quotes, check_redundant_parenthesis]: |
| 474 checker.name = 'eyeo' | 493 checker.name = 'eyeo' |
| 475 checker.version = __version__ | 494 checker.version = __version__ |
| OLD | NEW |