"""Adjust some old Python 2 idioms to their modern counterparts. * Change some type comparisons to isinstance() calls: type(x) == T -> isinstance(x, T) type(x) is T -> isinstance(x, T) type(x) != T -> not isinstance(x, T) type(x) is not T -> not isinstance(x, T) * Change "while 1:" into "while True:". * Change both v = list(EXPR) v.sort() foo(v) and the more general v = EXPR v.sort() foo(v) into v = sorted(EXPR) foo(v) """ # Author: Jacques Frechet, Collin Winter # Local imports from .. import fixer_base from ..fixer_util import Call, Comma, Name, Node, BlankLine, syms CMP = "(n='!=' | '==' | 'is' | n=comp_op< 'is' 'not' >)" TYPE = "power< 'type' trailer< '(' x=any ')' > >" class FixIdioms(fixer_base.BaseFix): explicit = True # The user must ask for this fixer PATTERN = r""" isinstance=comparison< %s %s T=any > | isinstance=comparison< T=any %s %s > | while_stmt< 'while' while='1' ':' any+ > | sorted=any< any* simple_stmt< expr_stmt< id1=any '=' power< list='list' trailer< '(' (not arglist<any+>) any ')' > > > '\n' > sort= simple_stmt< power< id2=any trailer< '.' 'sort' > trailer< '(' ')' > > '\n' > next=any* > | sorted=any< any* simple_stmt< expr_stmt< id1=any '=' expr=any > '\n' > sort= simple_stmt< power< id2=any trailer< '.' 'sort' > trailer< '(' ')' > > '\n' > next=any* > """ % (TYPE, CMP, CMP, TYPE) def match(self, node): r = super(FixIdioms, self).match(node) # If we've matched one of the sort/sorted subpatterns above, we # want to reject matches where the initial assignment and the # subsequent .sort() call involve different identifiers. if r and "sorted" in r: if r["id1"] == r["id2"]: return r return None return r def transform(self, node, results): if "isinstance" in results: return self.transform_isinstance(node, results) elif "while" in results: return self.transform_while(node, results) elif "sorted" in results: return self.transform_sort(node, results) else: raise RuntimeError("Invalid match") def transform_isinstance(self, node, results): x = results["x"].clone() # The thing inside of type() T = results["T"].clone() # The type being compared against x.prefix = "" T.prefix = " " test = Call(Name("isinstance"), [x, Comma(), T]) if "n" in results: test.prefix = " " test = Node(syms.not_test, [Name("not"), test]) test.prefix = node.prefix return test def transform_while(self, node, results): one = results["while"] one.replace(Name("True", prefix=one.prefix)) def transform_sort(self, node, results): sort_stmt = results["sort"] next_stmt = results["next"] list_call = results.get("list") simple_expr = results.get("expr") if list_call: list_call.replace(Name("sorted", prefix=list_call.prefix)) elif simple_expr: new = simple_expr.clone() new.prefix = "" simple_expr.replace(Call(Name("sorted"), [new], prefix=simple_expr.prefix)) else: raise RuntimeError("should not have reached here") sort_stmt.remove() btwn = sort_stmt.prefix # Keep any prefix lines between the sort_stmt and the list_call and # shove them right after the sorted() call. if "\n" in btwn: if next_stmt: # The new prefix should be everything from the sort_stmt's # prefix up to the last newline, then the old prefix after a new # line. prefix_lines = (btwn.rpartition("\n")[0], next_stmt[0].prefix) next_stmt[0].prefix = "\n".join(prefix_lines) else: assert list_call.parent assert list_call.next_sibling is None # Put a blank line after list_call and set its prefix. end_line = BlankLine() list_call.parent.append_child(end_line) assert list_call.next_sibling is end_line # The new prefix should be everything up to the first new line # of sort_stmt's prefix. end_line.prefix = btwn.rpartition("\n")[0]
Name | Type | Size | Permission | Actions |
---|---|---|---|---|
__pycache__ | Folder | 0755 |
|
|
__init__.py | File | 47 B | 0644 |
|
fix_apply.py | File | 2.29 KB | 0644 |
|
fix_asserts.py | File | 984 B | 0644 |
|
fix_basestring.py | File | 320 B | 0644 |
|
fix_buffer.py | File | 590 B | 0644 |
|
fix_dict.py | File | 3.67 KB | 0644 |
|
fix_except.py | File | 3.27 KB | 0644 |
|
fix_exec.py | File | 979 B | 0644 |
|
fix_execfile.py | File | 2 KB | 0644 |
|
fix_exitfunc.py | File | 2.44 KB | 0644 |
|
fix_filter.py | File | 2.7 KB | 0644 |
|
fix_funcattrs.py | File | 644 B | 0644 |
|
fix_future.py | File | 547 B | 0644 |
|
fix_getcwdu.py | File | 451 B | 0644 |
|
fix_has_key.py | File | 3.12 KB | 0644 |
|
fix_idioms.py | File | 4.76 KB | 0644 |
|
fix_import.py | File | 3.18 KB | 0644 |
|
fix_imports.py | File | 5.55 KB | 0644 |
|
fix_imports2.py | File | 289 B | 0644 |
|
fix_input.py | File | 708 B | 0644 |
|
fix_intern.py | File | 1.12 KB | 0644 |
|
fix_isinstance.py | File | 1.57 KB | 0644 |
|
fix_itertools.py | File | 1.51 KB | 0644 |
|
fix_itertools_imports.py | File | 2.04 KB | 0644 |
|
fix_long.py | File | 476 B | 0644 |
|
fix_map.py | File | 3.55 KB | 0644 |
|
fix_metaclass.py | File | 8 KB | 0644 |
|
fix_methodattrs.py | File | 606 B | 0644 |
|
fix_ne.py | File | 571 B | 0644 |
|
fix_next.py | File | 3.1 KB | 0644 |
|
fix_nonzero.py | File | 591 B | 0644 |
|
fix_numliterals.py | File | 768 B | 0644 |
|
fix_operator.py | File | 3.35 KB | 0644 |
|
fix_paren.py | File | 1.2 KB | 0644 |
|
fix_print.py | File | 2.78 KB | 0644 |
|
fix_raise.py | File | 2.86 KB | 0644 |
|
fix_raw_input.py | File | 454 B | 0644 |
|
fix_reduce.py | File | 837 B | 0644 |
|
fix_reload.py | File | 1.06 KB | 0644 |
|
fix_renames.py | File | 2.17 KB | 0644 |
|
fix_repr.py | File | 613 B | 0644 |
|
fix_set_literal.py | File | 1.66 KB | 0644 |
|
fix_standarderror.py | File | 449 B | 0644 |
|
fix_sys_exc.py | File | 1.01 KB | 0644 |
|
fix_throw.py | File | 1.54 KB | 0644 |
|
fix_tuple_params.py | File | 5.43 KB | 0644 |
|
fix_types.py | File | 1.73 KB | 0644 |
|
fix_unicode.py | File | 1.23 KB | 0644 |
|
fix_urllib.py | File | 8.16 KB | 0644 |
|
fix_ws_comma.py | File | 1.06 KB | 0644 |
|
fix_xrange.py | File | 2.63 KB | 0644 |
|
fix_xreadlines.py | File | 689 B | 0644 |
|
fix_zip.py | File | 1.26 KB | 0644 |
|