summaryrefslogtreecommitdiffstats
path: root/Lib/lib2to3/fixes/fix_imports.py
blob: 151eb1c34372939b667574ab9080759a14820ff4 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
"""Fix incompatible imports and module references."""
# Author: Collin Winter

# Local imports
from .. import fixer_base
from ..fixer_util import Name, attr_chain, any, set
import builtins
builtin_names = [name for name in dir(builtins)
                 if name not in ("__name__", "__doc__", "exec", "print")]

# XXX(alexandre): It would be possible to get the modules exports by fetching
# XXX: their __all__ attribute. However, I fear that this would add an additional
# XXX: overhead to the fixer.
MAPPING = {"StringIO":  ("io", ["StringIO"]),
           "cStringIO": ("io", ["StringIO"]),
           "cPickle": ("pickle", ['BadPickleGet', 'HIGHEST_PROTOCOL',
                                  'PickleError', 'Pickler', 'PicklingError',
                                  'UnpickleableError', 'Unpickler', 'UnpicklingError',
                                  'compatible_formats', 'dump', 'dumps', 'format_version',
                                  'load', 'loads']),
           "__builtin__" : ("builtins", builtin_names),
           'copy_reg': ('copyreg', ['pickle',
                                    'constructor',
                                    'add_extension',
                                    'remove_extension',
                                    'clear_extension_cache']),
           'Queue': ('queue', ['Empty', 'Full', 'Queue',
                               'PriorityQueue', 'LifoQueue']),
           'SocketServer': ('socketserver',
                            ['TCPServer', 'UDPServer', 'BaseServer',
                             'ForkingUDPServer', 'ForkingTCPServer',
                             'ThreadingUDPServer', 'ThreadingTCPServer',
                             'BaseRequestHandler', 'StreamRequestHandler',
                             'DatagramRequestHandler', 'ThreadingMixIn',
                             'ForkingMixIn', 'UnixStreamServer',
                             'UnixDatagramServer', 'ThreadingUnixStreamServer',
                             'ThreadingUnixDatagramServer']),
           'ConfigParser': ('configparser',
                            ['NoSectionError', 'DuplicateSectionError',
                             'NoOptionError', 'InterpolationError',
                             'InterpolationDepthError',
                             'InterpolationSyntaxError',
                             'ParsingError', 'MissingSectionHeaderError',
                             'ConfigParser', 'SafeConfigParser',
                             'RawConfigParser', 'DEFAULTSECT',
                             'MAX_INTERPOLATION_DEPTH']),
           'repr': ('reprlib', ['Repr', 'repr']),
           'FileDialog': ('tkinter.filedialog',
                          ['FileDialog', 'LoadFileDialog', 'SaveFileDialog']),
           'tkFileDialog': ('tkinter.filedialog',
                            ['Open', 'SaveAs', 'Directory', 'askopenfilename',
                             'asksaveasfilename', 'askopenfilenames',
                             'askopenfile', 'askopenfiles', 'asksaveasfile',
                             'askdirectory']),
           'SimpleDialog': ('tkinter.simpledialog', ['SimpleDialog']),
           'tkSimpleDialog': ('tkinter.simpledialog',
                              ['Dialog', 'askinteger', 'askfloat',
                               'askstring']),
           'tkColorChooser': ('tkinter.colorchooser', ['Chooser', 'askcolor']),
           'tkCommonDialog': ('tkinter.commondialog', ['Dialog']),
           'Dialog': ('tkinter.dialog', ['Dialog']),
           'Tkdnd': ('tkinter.dnd', ['DndHandler']),
           'tkFont': ('tkinter.font',
                      ['nametofont', 'Font', 'families', 'names']),
           'tkMessageBox': ('tkinter.messagebox',
                            ['Message', 'showinfo', 'showwarning', 'showerror',
                             'askquestion', 'askokcancel', 'askyesno',
                             'askyesnocancel', 'askretrycancel']),
           'ScrolledText': ('tkinter.scrolledtext', ['ScrolledText']),
           'turtle': ('tkinter.turtle',
                      ['RawPen', 'Pen', 'Turtle', 'degrees', 'radian', 'reset',
                       'clear', 'tracer', 'forward', 'backward', 'left',
                       'right', 'up', 'down', 'width', 'color', 'write', 'fill',
                       'begin_fill', 'end_fill', 'circle', 'goto', 'heading',
                       'setheading', 'position', 'window_width', 'setx', 'sety',
                       'towards', 'done', 'delay', 'speed', 'setup', 'title']),
           'Tkconstants': ('tkinter.constants',
                           ['NO', 'FALSE', 'OFF', 'YES', 'TRUE', 'ON', 'N', 'S',
                            'W', 'E', 'NW', 'SW', 'SE', 'NE', 'NS', 'EW',
                            'NSEW', 'CENTER', 'NONE', 'X', 'Y', 'BOTH', 'LEFT',
                            'TOP', 'RIGHT', 'BOTTOM', 'RAISED', 'SUNKEN',
                            'FLAT', 'RIDGE', 'GROOVE', 'SOLID', 'HORIZONTAL',
                            'VERTICAL', 'NUMERIC', 'CHAR', 'WORD', 'BASELINE',
                            'INSIDE', 'OUTSIDE', 'SEL', 'SEL_FIRST', 'SEL_LAST',
                            'END', 'INSERT', 'CURRENT', 'ANCHOR', 'ALL',
                            'NORMAL', 'DISABLED', 'ACTIVE', 'HIDDEN', 'CASCADE',
                            'CHECKBUTTON', 'COMMAND', 'RADIOBUTTON',
                            'SEPARATOR', 'SINGLE', 'BROWSE', 'MULTIPLE',
                            'EXTENDED', 'DOTBOX', 'UNDERLINE', 'PIESLICE',
                            'CHORD', 'ARC', 'FIRST', 'LAST', 'BUTT',
                            'PROJECTING', 'ROUND', 'BEVEL', 'MITTER', 'MOVETO',
                            'SCROLL', 'UNITS', 'PAGES']),
           'Tix': ('tkinter.tix',
                   ['tixCommand', 'Tk', 'Form', 'TixWidget', 'TixSubWidget',
                    'DisplayStyle', 'Balloon', 'ButtonBox', 'ComboBox',
                    'Control', 'DirList', 'DirTree', 'DirSelectBox',
                    'ExFileSelectBox', 'DirSelectDialog', 'ExFileSelectDialog',
                    'FileSelectBox', 'FileSelectDialog', 'FileEntry', 'HList',
                    'InputOnly', 'LabelEntry', 'LabelFrame', 'ListNoteBook',
                    'Meter', 'NoteBook', 'OptionMenu', 'PanedWindow',
                    'PopupMenu', 'ResizeHandle', 'ScrolledHList',
                    'ScrolledListBox', 'ScrolledText', 'ScrolledTList',
                    'ScrolledWindow', 'Select', 'Shell', 'DialogShell',
                    'StdButtonBox', 'TList', 'Tree', 'CheckList', 'OptionName',
                    'FileTypeList', 'Grid', 'ScrolledGrid']),
           'Tkinter': ('tkinter',
                       ['_flatten', 'TclError', 'TkVersion', 'TclVersion',
                        'Variable', 'StringVar', 'IntVar', 'DoubleVar',
                        'BooleanVar','mainloop', 'Tk', 'Tcl', 'Toplevel',
                        'Button', 'Canvas', 'Checkbutton', 'Entry', 'Frame',
                        'Label', 'Listbox', 'Menu', 'Menubutton',
                        'Radiobutton', 'Scale', 'Scrollbar', 'Text',
                        'OptionMenu', 'Image', 'PhotoImage', 'BitmapImage',
                        'image_names', 'image_types', 'Spinbox', 'LabelFrame',
                        'PanedWindow', 'Studbutton', 'Tributton']),
           'markupbase': ('_markupbase', ['ParserBase']),
           '_winreg': ('winreg', [
               'CloseKey', 'ConnectRegistry', 'CreateKey', 'DeleteKey',
               'DeleteValue', 'DisableReflectionKey', 'EnableReflectionKey',
               'EnumKey', 'EnumValue', 'ExpandEnvironmentStrings', 'FlushKey',
               'LoadKey', 'OpenKey', 'OpenKeyEx', 'QueryValue', 'QueryValueEx',
               'QueryInfoKey', 'QueryReflectionKey', 'SaveKey', 'SetValue',
               'SetValueEx', 'HKEY_CLASSES_ROOT', 'HKEY_CURRENT_USER',
               'HKEY_LOCAL_MACHINE', 'HKEY_USERS', 'HKEY_PERFORMANCE_DATA',
               'HKEY_CURRENT_CONFIG', 'HKEY_DYN_DATA', 'KEY_QUERY_VALUE',
               'KEY_SET_VALUE', 'KEY_CREATE_SUB_KEY', 'KEY_ENUMERATE_SUB_KEYS',
               'KEY_NOTIFY', 'KEY_CREATE_LINK', 'KEY_READ', 'KEY_WRITE',
               'KEY_EXECUTE', 'KEY_ALL_ACCESS', 'KEY_WOW64_64KEY',
               'KEY_WOW64_32KEY', 'REG_OPTION_RESERVED',
               'REG_OPTION_NON_VOLATILE', 'REG_OPTION_VOLATILE',
               'REG_OPTION_CREATE_LINK', 'REG_OPTION_BACKUP_RESTORE',
               'REG_OPTION_OPEN_LINK', 'REG_LEGAL_OPTION',
               'REG_CREATED_NEW_KEY', 'REG_OPENED_EXISTING_KEY',
               'REG_WHOLE_HIVE_VOLATILE', 'REG_REFRESH_HIVE',
               'REG_NO_LAZY_FLUSH', 'REG_NOTIFY_CHANGE_NAME',
               'REG_NOTIFY_CHANGE_ATTRIBUTES', 'REG_NOTIFY_CHANGE_LAST_SET',
               'REG_NOTIFY_CHANGE_SECURITY', 'REG_LEGAL_CHANGE_FILTER',
               'REG_NONE', 'REG_SZ', 'REG_EXPAND_SZ', 'REG_BINARY', 'REG_DWORD',
               'REG_DWORD_LITTLE_ENDIAN', 'REG_DWORD_BIG_ENDIAN', 'REG_LINK',
               'REG_MULTI_SZ', 'REG_RESOURCE_LIST',
               'REG_FULL_RESOURCE_DESCRIPTOR', 'REG_RESOURCE_REQUIREMENTS_LIST']),
           'thread': ('_thread',
                      ['LockType', '_local', 'allocate', 'allocate_lock',
                       'error', 'exit', 'exit_thread', 'get_ident',
                       'interrupt_main', 'stack_size', 'start_new',
                       'start_new_thread']),
           'dummy_thread': ('_dummy_thread',
                      ['LockType', '_local', 'allocate', 'allocate_lock',
                       'error', 'exit', 'exit_thread', 'get_ident',
                       'interrupt_main', 'stack_size', 'start_new',
                       'start_new_thread']),
           # anydbm and whichdb are handed by fix_imports2.
           'dbhash': ('dbm.bsd', ['error', 'open']),
           'dumbdbm': ('dbm.dumb', ['error', 'open', '_Database']),
           'dbm': ('dbm.ndbm', ['error', 'open', 'library']),
           'gdbm': ('dbm.gnu', ['error', 'open', 'open_flags']),
           'xmlrpclib': ('xmlrpc.client',
                         ['Error', 'ProtocolError', 'ResponseError', 'Fault',
                          'ServerProxy', 'Boolean', 'DateTime', 'Binary',
                          'ExpatParser', 'FastMarshaller', 'FastParser',
                          'FastUnmarshaller', 'MultiCall', 'MultiCallIterator',
                          'SlowParser', 'Marshaller', 'Unmarshaller', 'Server',
                          'Transport', 'SafeTransport', 'SgmlopParser',
                          'boolean', 'getparser', 'dumps', 'loads', 'escape',
                          'PARSE_ERROR', 'SERVER_ERROR', 'WRAPPERS',
                          'APPLICATION_ERROR', 'SYSTEM_ERROR',
                          'TRANSPORT_ERROR', 'NOT_WELLFORMED_ERROR',
                          'UNSUPPORTED_ENCODING', 'INVALID_ENCODING_CHAR',
                          'INVALID_XMLRPC', 'METHOD_NOT_FOUND',
                          'INVALID_METHOD_PARAMS', 'INTERNAL_ERROR',
                          'MININT', 'MAXINT']),
           'DocXMLRPCServer': ('xmlrpc.server',
                               ['CGIXMLRPCRequestHandler',
                               'DocCGIXMLRPCRequestHandler',
                               'DocXMLRPCRequestHandler', 'DocXMLRPCServer',
                               'ServerHTMLDoc', 'SimpleXMLRPCRequestHandler',
                               'SimpleXMLRPCServer', 'XMLRPCDocGenerator',
                               'resolve_dotted_attribute']),
           'SimpleXMLRPCServer': ('xmlrpc.server',
                                  ['CGIXMLRPCRequestHandler',
                                   'Fault', 'SimpleXMLRPCDispatcher',
                                   'SimpleXMLRPCRequestHandler',
                                   'SimpleXMLRPCServer', 'SocketServer',
                                   'list_public_methods',
                                   'remove_duplicates',
                                   'resolve_dotted_attribute']),
           'httplib': ('http.client',
                       ['ACCEPTED', 'BAD_GATEWAY', 'BAD_REQUEST',
                        'BadStatusLine', 'CONFLICT', 'CONTINUE', 'CREATED',
                        'CannotSendHeader', 'CannotSendRequest',
                        'EXPECTATION_FAILED', 'FAILED_DEPENDENCY', 'FORBIDDEN',
                        'FOUND', 'FakeSocket', 'GATEWAY_TIMEOUT', 'GONE',
                        'HTTP', 'HTTPConnection', 'HTTPException',
                        'HTTPMessage', 'HTTPResponse', 'HTTPS',
                        'HTTPSConnection', 'HTTPS_PORT', 'HTTP_PORT',
                        'HTTP_VERSION_NOT_SUPPORTED', 'IM_USED',
                        'INSUFFICIENT_STORAGE', 'INTERNAL_SERVER_ERROR',
                        'ImproperConnectionState', 'IncompleteRead',
                        'InvalidURL', 'LENGTH_REQUIRED', 'LOCKED',
                        'LineAndFileWrapper', 'MAXAMOUNT', 'METHOD_NOT_ALLOWED',
                        'MOVED_PERMANENTLY', 'MULTIPLE_CHOICES', 'MULTI_STATUS',
                        'NON_AUTHORITATIVE_INFORMATION', 'NOT_ACCEPTABLE',
                        'NOT_EXTENDED', 'NOT_FOUND', 'NOT_IMPLEMENTED',
                        'NOT_MODIFIED', 'NO_CONTENT', 'NotConnected', 'OK',
                        'PARTIAL_CONTENT', 'PAYMENT_REQUIRED',
                        'PRECONDITION_FAILED', 'PROCESSING',
                        'PROXY_AUTHENTICATION_REQUIRED',
                        'REQUESTED_RANGE_NOT_SATISFIABLE',
                        'REQUEST_ENTITY_TOO_LARGE', 'REQUEST_TIMEOUT',
                        'REQUEST_URI_TOO_LONG', 'RESET_CONTENT',
                        'ResponseNotReady', 'SEE_OTHER', 'SERVICE_UNAVAILABLE',
                        'SSLFile', 'SWITCHING_PROTOCOLS', 'SharedSocket',
                        'SharedSocketClient', 'StringIO', 'TEMPORARY_REDIRECT',
                        'UNAUTHORIZED', 'UNPROCESSABLE_ENTITY',
                        'UNSUPPORTED_MEDIA_TYPE', 'UPGRADE_REQUIRED',
                        'USE_PROXY', 'UnimplementedFileMode', 'UnknownProtocol',
                        'UnknownTransferEncoding', 'error', 'responses']),
           'Cookie': ('http.cookies',
                      ['BaseCookie', 'Cookie', 'CookieError', 'Morsel',
                       'SerialCookie', 'SimpleCookie', 'SmartCookie']),
           'cookielib': ('http.cookiejar',
                         ['Absent', 'Cookie', 'CookieJar', 'CookiePolicy',
                          'DAYS', 'DEFAULT_HTTP_PORT', 'DefaultCookiePolicy',
                          'EPOCH_YEAR', 'ESCAPED_CHAR_RE', 'FileCookieJar',
                          'HEADER_ESCAPE_RE', 'HEADER_JOIN_ESCAPE_RE',
                          'HEADER_QUOTED_VALUE_RE', 'HEADER_TOKEN_RE',
                          'HEADER_VALUE_RE', 'HTTP_PATH_SAFE', 'IPV4_RE',
                          'ISO_DATE_RE', 'LOOSE_HTTP_DATE_RE', 'LWPCookieJar',
                          'LoadError', 'MISSING_FILENAME_TEXT', 'MONTHS',
                          'MONTHS_LOWER', 'MozillaCookieJar', 'STRICT_DATE_RE',
                          'TIMEZONE_RE', 'UTC_ZONES', 'WEEKDAY_RE',
                          'cut_port_re', 'deepvalues', 'domain_match',
                          'eff_request_host', 'escape_path', 'http2time',
                          'is_HDN', 'is_third_party', 'iso2time',
                          'join_header_words', 'liberal_is_HDN', 'logger',
                          'lwp_cookie_str', 'month', 'offset_from_tz_string',
                          'parse_ns_headers', 'reach', 'request_host',
                          'request_path', 'request_port', 'split_header_words',
                          'time', 'time2isoz', 'time2netscape', 'unmatched',
                          'uppercase_escaped_char', 'urllib',
                          'user_domain_match', 'vals_sorted_by_key']),
           'BaseHTTPServer': ('http.server',
                              ['BaseHTTPRequestHandler',
                               'DEFAULT_ERROR_MESSAGE', 'HTTPServer']),
           'SimpleHTTPServer': ('http.server', ['SimpleHTTPRequestHandler']),
           'CGIHTTPServer': ('http.server',
                             ['CGIHTTPRequestHandler', 'executable',
                              'nobody_uid', 'nobody']),
           # 'test.test_support': ('test.support',
           #                ["Error", "TestFailed", "TestSkipped", "ResourceDenied",
           #                "import_module", "verbose", "use_resources",
           #                "max_memuse", "record_original_stdout",
           #                "get_original_stdout", "unload", "unlink", "rmtree",
           #                "forget", "is_resource_enabled", "requires",
           #                "find_unused_port", "bind_port",
           #                "fcmp", "is_jython", "TESTFN", "HOST",
           #                "FUZZ", "findfile", "verify", "vereq", "sortdict",
           #                "check_syntax_error", "open_urlresource", "WarningMessage",
           #                "catch_warning", "CleanImport", "EnvironmentVarGuard",
           #                "TransientResource", "captured_output", "captured_stdout",
           #                "TransientResource", "transient_internet", "run_with_locale",
           #                "set_memlimit", "bigmemtest", "bigaddrspacetest",
           #                "BasicTestRunner", "run_unittest", "run_doctest",
           #                "threading_setup", "threading_cleanup", "reap_children"]),
           'commands': ('subprocess', ['getstatusoutput', 'getoutput']),
           'UserString' : ('collections', ['UserString']),
           'UserList' : ('collections', ['UserList']),
           'urlparse' : ('urllib.parse',
                            ['urlparse', 'urlunparse', 'urlsplit',
                            'urlunsplit', 'urljoin', 'urldefrag']),
            'robotparser' : ('urllib.robotparser', ['RobotFileParser']),
}


def alternates(members):
    return "(" + "|".join(map(repr, members)) + ")"


def build_pattern(mapping=MAPPING):
    bare = set()
    for old_module, (new_module, members) in list(mapping.items()):
        bare.add(old_module)
        bare.update(members)
        members = alternates(members)
        yield """import_name< 'import' (module=%r
                              | dotted_as_names< any* module=%r any* >) >
              """ % (old_module, old_module)
        yield """import_from< 'from' module_name=%r 'import'
                   ( %s | import_as_name< %s 'as' any > |
                     import_as_names< any* >) >
              """ % (old_module, members, members)
        yield """import_from< 'from' module_name=%r 'import' star='*' >
              """ % old_module
        yield """import_name< 'import'
                              dotted_as_name< module_name=%r 'as' any > >
              """ % old_module
        # Find usages of module members in code e.g. urllib.foo(bar)
        yield """power< module_name=%r trailer< '.' %s > any* >
              """ % (old_module, members)
    yield """bare_name=%s""" % alternates(bare)


class FixImports(fixer_base.BaseFix):
    PATTERN = "|".join(build_pattern())

    order = "pre" # Pre-order tree traversal

    mapping = MAPPING

    # Don't match the node if it's within another match
    def match(self, node):
        match = super(FixImports, self).match
        results = match(node)
        if results:
            if any([match(obj) for obj in attr_chain(node, "parent")]):
                return False
            return results
        return False

    def start_tree(self, tree, filename):
        super(FixImports, self).start_tree(tree, filename)
        self.replace = {}

    def transform(self, node, results):
        import_mod = results.get("module")
        mod_name = results.get("module_name")
        bare_name = results.get("bare_name")
        star = results.get("star")

        if import_mod or mod_name:
            new_name, members = self.mapping[(import_mod or mod_name).value]

        if import_mod:
            self.replace[import_mod.value] = new_name
            import_mod.replace(Name(new_name, prefix=import_mod.get_prefix()))
        elif mod_name:
            if star:
                self.cannot_convert(node, "Cannot handle star imports.")
            else:
                mod_name.replace(Name(new_name, prefix=mod_name.get_prefix()))
        elif bare_name:
            bare_name = bare_name[0]
            new_name = self.replace.get(bare_name.value)
            if new_name:
                bare_name.replace(Name(new_name, prefix=bare_name.get_prefix()))