##// END OF EJS Templates
demandimport: fix import x.y.z as a when x.y is already imported.
Matt Mackall -
r3921:6d0d025e default
parent child Browse files
Show More
@@ -1,109 +1,115
1 1 # demandimport.py - global demand-loading of modules for Mercurial
2 2 #
3 3 # Copyright 2006 Matt Mackall <mpm@selenic.com>
4 4 #
5 5 # This software may be used and distributed according to the terms
6 6 # of the GNU General Public License, incorporated herein by reference.
7 7
8 8 '''
9 9 demandimport - automatic demandloading of modules
10 10
11 11 To enable this module, do:
12 12
13 13 import demandimport; demandimport.enable()
14 14
15 15 Imports of the following forms will be demand-loaded:
16 16
17 17 import a, b.c
18 18 import a.b as c
19 19 from a import b,c # a will be loaded immediately
20 20
21 21 These imports will not be delayed:
22 22
23 23 from a import *
24 24 b = __import__(a)
25 25 '''
26 26
27 27 _origimport = __import__
28 28
29 29 class _demandmod(object):
30 30 """module demand-loader and proxy"""
31 31 def __init__(self, name, globals, locals):
32 32 if '.' in name:
33 33 head, rest = name.split('.', 1)
34 34 after = [rest]
35 35 else:
36 36 head = name
37 37 after = []
38 38 object.__setattr__(self, "_data", (head, globals, locals, after))
39 39 object.__setattr__(self, "_module", None)
40 40 def _extend(self, name):
41 41 """add to the list of submodules to load"""
42 42 self._data[3].append(name)
43 43 def _load(self):
44 44 if not self._module:
45 45 head, globals, locals, after = self._data
46 46 mod = _origimport(head, globals, locals)
47 47 # load submodules
48 def subload(mod, p):
49 h, t = p, None
50 if '.' in p:
51 h, t = p.split('.', 1)
52 if not hasattr(mod, h):
53 setattr(mod, h, _demandmod(p, mod.__dict__, mod.__dict__))
54 else:
55 subload(getattr(mod, h), t)
56
48 57 for x in after:
49 hx = x
50 if '.' in x:
51 hx = x.split('.')[0]
52 if not hasattr(mod, hx):
53 setattr(mod, hx, _demandmod(x, mod.__dict__, mod.__dict__))
58 subload(mod, x)
59
54 60 # are we in the locals dictionary still?
55 61 if locals and locals.get(head) == self:
56 62 locals[head] = mod
57 63 object.__setattr__(self, "_module", mod)
58 64 def __repr__(self):
59 65 return "<unloaded module '%s'>" % self._data[0]
60 66 def __call__(self, *args, **kwargs):
61 67 raise TypeError("'unloaded module' object is not callable")
62 68 def __getattribute__(self, attr):
63 69 if attr in ('_data', '_extend', '_load', '_module'):
64 70 return object.__getattribute__(self, attr)
65 71 self._load()
66 72 return getattr(self._module, attr)
67 73 def __setattr__(self, attr, val):
68 74 self._load()
69 75 setattr(self._module, attr, val)
70 76
71 77 def _demandimport(name, globals=None, locals=None, fromlist=None):
72 78 if not locals or name in ignore or fromlist == ('*',):
73 79 # these cases we can't really delay
74 80 return _origimport(name, globals, locals, fromlist)
75 81 elif not fromlist:
76 82 # import a [as b]
77 83 if '.' in name: # a.b
78 84 base, rest = name.split('.', 1)
79 85 # email.__init__ loading email.mime
80 86 if globals and globals.get('__name__', None) == base:
81 87 return _origimport(name, globals, locals, fromlist)
82 88 # if a is already demand-loaded, add b to its submodule list
83 89 if base in locals:
84 90 if isinstance(locals[base], _demandmod):
85 91 locals[base]._extend(rest)
86 92 return locals[base]
87 93 return _demandmod(name, globals, locals)
88 94 else:
89 95 # from a import b,c,d
90 96 mod = _origimport(name, globals, locals)
91 97 # recurse down the module chain
92 98 for comp in name.split('.')[1:]:
93 99 mod = getattr(mod, comp)
94 100 for x in fromlist:
95 101 # set requested submodules for demand load
96 102 if not(hasattr(mod, x)):
97 103 setattr(mod, x, _demandmod(x, mod.__dict__, mod.__dict__))
98 104 return mod
99 105
100 106 ignore = []
101 107
102 108 def enable():
103 109 "enable global demand-loading of modules"
104 110 __builtins__["__import__"] = _demandimport
105 111
106 112 def disable():
107 113 "disable global demand-loading of modules"
108 114 __builtins__["__import__"] = _origimport
109 115
General Comments 0
You need to be logged in to leave comments. Login now