Mercurial > repos > guerler > springsuite
comparison planemo/lib/python3.7/site-packages/libpasteurize/fixes/fix_kwargs.py @ 1:56ad4e20f292 draft
"planemo upload commit 6eee67778febed82ddd413c3ca40b3183a3898f1"
author | guerler |
---|---|
date | Fri, 31 Jul 2020 00:32:28 -0400 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
0:d30785e31577 | 1:56ad4e20f292 |
---|---|
1 u""" | |
2 Fixer for Python 3 function parameter syntax | |
3 This fixer is rather sensitive to incorrect py3k syntax. | |
4 """ | |
5 | |
6 # Note: "relevant" parameters are parameters following the first STAR in the list. | |
7 | |
8 from lib2to3 import fixer_base | |
9 from lib2to3.fixer_util import token, String, Newline, Comma, Name | |
10 from libfuturize.fixer_util import indentation, suitify, DoubleStar | |
11 | |
12 _assign_template = u"%(name)s = %(kwargs)s['%(name)s']; del %(kwargs)s['%(name)s']" | |
13 _if_template = u"if '%(name)s' in %(kwargs)s: %(assign)s" | |
14 _else_template = u"else: %(name)s = %(default)s" | |
15 _kwargs_default_name = u"_3to2kwargs" | |
16 | |
17 def gen_params(raw_params): | |
18 u""" | |
19 Generator that yields tuples of (name, default_value) for each parameter in the list | |
20 If no default is given, then it is default_value is None (not Leaf(token.NAME, 'None')) | |
21 """ | |
22 assert raw_params[0].type == token.STAR and len(raw_params) > 2 | |
23 curr_idx = 2 # the first place a keyword-only parameter name can be is index 2 | |
24 max_idx = len(raw_params) | |
25 while curr_idx < max_idx: | |
26 curr_item = raw_params[curr_idx] | |
27 prev_item = curr_item.prev_sibling | |
28 if curr_item.type != token.NAME: | |
29 curr_idx += 1 | |
30 continue | |
31 if prev_item is not None and prev_item.type == token.DOUBLESTAR: | |
32 break | |
33 name = curr_item.value | |
34 nxt = curr_item.next_sibling | |
35 if nxt is not None and nxt.type == token.EQUAL: | |
36 default_value = nxt.next_sibling | |
37 curr_idx += 2 | |
38 else: | |
39 default_value = None | |
40 yield (name, default_value) | |
41 curr_idx += 1 | |
42 | |
43 def remove_params(raw_params, kwargs_default=_kwargs_default_name): | |
44 u""" | |
45 Removes all keyword-only args from the params list and a bare star, if any. | |
46 Does not add the kwargs dict if needed. | |
47 Returns True if more action is needed, False if not | |
48 (more action is needed if no kwargs dict exists) | |
49 """ | |
50 assert raw_params[0].type == token.STAR | |
51 if raw_params[1].type == token.COMMA: | |
52 raw_params[0].remove() | |
53 raw_params[1].remove() | |
54 kw_params = raw_params[2:] | |
55 else: | |
56 kw_params = raw_params[3:] | |
57 for param in kw_params: | |
58 if param.type != token.DOUBLESTAR: | |
59 param.remove() | |
60 else: | |
61 return False | |
62 else: | |
63 return True | |
64 | |
65 def needs_fixing(raw_params, kwargs_default=_kwargs_default_name): | |
66 u""" | |
67 Returns string with the name of the kwargs dict if the params after the first star need fixing | |
68 Otherwise returns empty string | |
69 """ | |
70 found_kwargs = False | |
71 needs_fix = False | |
72 | |
73 for t in raw_params[2:]: | |
74 if t.type == token.COMMA: | |
75 # Commas are irrelevant at this stage. | |
76 continue | |
77 elif t.type == token.NAME and not found_kwargs: | |
78 # Keyword-only argument: definitely need to fix. | |
79 needs_fix = True | |
80 elif t.type == token.NAME and found_kwargs: | |
81 # Return 'foobar' of **foobar, if needed. | |
82 return t.value if needs_fix else u'' | |
83 elif t.type == token.DOUBLESTAR: | |
84 # Found either '*' from **foobar. | |
85 found_kwargs = True | |
86 else: | |
87 # Never found **foobar. Return a synthetic name, if needed. | |
88 return kwargs_default if needs_fix else u'' | |
89 | |
90 class FixKwargs(fixer_base.BaseFix): | |
91 | |
92 run_order = 7 # Run after function annotations are removed | |
93 | |
94 PATTERN = u"funcdef< 'def' NAME parameters< '(' arglist=typedargslist< params=any* > ')' > ':' suite=any >" | |
95 | |
96 def transform(self, node, results): | |
97 params_rawlist = results[u"params"] | |
98 for i, item in enumerate(params_rawlist): | |
99 if item.type == token.STAR: | |
100 params_rawlist = params_rawlist[i:] | |
101 break | |
102 else: | |
103 return | |
104 # params is guaranteed to be a list starting with *. | |
105 # if fixing is needed, there will be at least 3 items in this list: | |
106 # [STAR, COMMA, NAME] is the minimum that we need to worry about. | |
107 new_kwargs = needs_fixing(params_rawlist) | |
108 # new_kwargs is the name of the kwargs dictionary. | |
109 if not new_kwargs: | |
110 return | |
111 suitify(node) | |
112 | |
113 # At this point, params_rawlist is guaranteed to be a list | |
114 # beginning with a star that includes at least one keyword-only param | |
115 # e.g., [STAR, NAME, COMMA, NAME, COMMA, DOUBLESTAR, NAME] or | |
116 # [STAR, COMMA, NAME], or [STAR, COMMA, NAME, COMMA, DOUBLESTAR, NAME] | |
117 | |
118 # Anatomy of a funcdef: ['def', 'name', parameters, ':', suite] | |
119 # Anatomy of that suite: [NEWLINE, INDENT, first_stmt, all_other_stmts] | |
120 # We need to insert our new stuff before the first_stmt and change the | |
121 # first_stmt's prefix. | |
122 | |
123 suite = node.children[4] | |
124 first_stmt = suite.children[2] | |
125 ident = indentation(first_stmt) | |
126 | |
127 for name, default_value in gen_params(params_rawlist): | |
128 if default_value is None: | |
129 suite.insert_child(2, Newline()) | |
130 suite.insert_child(2, String(_assign_template %{u'name':name, u'kwargs':new_kwargs}, prefix=ident)) | |
131 else: | |
132 suite.insert_child(2, Newline()) | |
133 suite.insert_child(2, String(_else_template %{u'name':name, u'default':default_value}, prefix=ident)) | |
134 suite.insert_child(2, Newline()) | |
135 suite.insert_child(2, String(_if_template %{u'assign':_assign_template %{u'name':name, u'kwargs':new_kwargs}, u'name':name, u'kwargs':new_kwargs}, prefix=ident)) | |
136 first_stmt.prefix = ident | |
137 suite.children[2].prefix = u"" | |
138 | |
139 # Now, we need to fix up the list of params. | |
140 | |
141 must_add_kwargs = remove_params(params_rawlist) | |
142 if must_add_kwargs: | |
143 arglist = results[u'arglist'] | |
144 if len(arglist.children) > 0 and arglist.children[-1].type != token.COMMA: | |
145 arglist.append_child(Comma()) | |
146 arglist.append_child(DoubleStar(prefix=u" ")) | |
147 arglist.append_child(Name(new_kwargs)) |