mirror of
				https://github.com/python/cpython.git
				synced 2025-10-31 13:41:24 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			126 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			126 lines
		
	
	
	
		
			3 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| """An Python re-implementation of hierarchical module import.
 | |
| 
 | |
| This code is intended to be read, not executed.  However, it does work
 | |
| -- all you need to do to enable it is "import knee".
 | |
| 
 | |
| (The name is a pun on the klunkier predecessor of this module, "ni".)
 | |
| 
 | |
| """
 | |
| 
 | |
| import sys, imp, __builtin__, string
 | |
| 
 | |
| 
 | |
| # Replacement for __import__()
 | |
| def import_hook(name, globals=None, locals=None, fromlist=None):
 | |
|     parent = determine_parent(globals)
 | |
|     q, tail = find_head_package(parent, name)
 | |
|     m = load_tail(q, tail)
 | |
|     if not fromlist:
 | |
| 	return q
 | |
|     if hasattr(m, "__path__"):
 | |
| 	ensure_fromlist(m, fromlist)
 | |
|     return m
 | |
| 
 | |
| def determine_parent(globals):
 | |
|     if not globals or  not globals.has_key("__name__"):
 | |
| 	return None
 | |
|     pname = globals['__name__']
 | |
|     if globals.has_key("__path__"):
 | |
| 	parent = sys.modules[pname]
 | |
| 	assert globals is parent.__dict__
 | |
| 	return parent
 | |
|     if '.' in pname:
 | |
| 	i = string.rfind(pname, '.')
 | |
| 	pname = pname[:i]
 | |
| 	parent = sys.modules[pname]
 | |
| 	assert parent.__name__ == pname
 | |
| 	return parent
 | |
|     return None
 | |
| 
 | |
| def find_head_package(parent, name):
 | |
|     if '.' in name:
 | |
| 	i = string.find(name, '.')
 | |
| 	head = name[:i]
 | |
| 	tail = name[i+1:]
 | |
|     else:
 | |
| 	head = name
 | |
| 	tail = ""
 | |
|     if parent:
 | |
| 	qname = "%s.%s" % (parent.__name__, head)
 | |
|     else:
 | |
| 	qname = head
 | |
|     q = import_module(head, qname, parent)
 | |
|     if q: return q, tail
 | |
|     if parent:
 | |
| 	qname = head
 | |
| 	parent = None
 | |
| 	q = import_module(head, qname, parent)
 | |
| 	if q: return q, tail
 | |
|     raise ImportError, "No module named " + qname
 | |
| 
 | |
| def load_tail(q, tail):
 | |
|     m = q
 | |
|     while tail:
 | |
| 	i = string.find(tail, '.')
 | |
| 	if i < 0: i = len(tail)
 | |
| 	head, tail = tail[:i], tail[i+1:]
 | |
| 	mname = "%s.%s" % (m.__name__, head)
 | |
| 	m = import_module(head, mname, m)
 | |
| 	if not m:
 | |
| 	    raise ImportError, "No module named " + mname
 | |
|     return m
 | |
| 
 | |
| def ensure_fromlist(m, fromlist, recursive=0):
 | |
|     for sub in fromlist:
 | |
| 	if sub == "*":
 | |
| 	    if not recursive:
 | |
| 		try:
 | |
| 		    all = m.__all__
 | |
| 		except AttributeError:
 | |
| 		    pass
 | |
| 		else:
 | |
| 		    ensure_fromlist(m, all, 1)
 | |
| 	    continue
 | |
| 	if sub != "*" and not hasattr(m, sub):
 | |
| 	    subname = "%s.%s" % (m.__name__, sub)
 | |
| 	    submod = import_module(sub, subname, m)
 | |
| 	    if not submod:
 | |
| 		raise ImportError, "No module named " + subname
 | |
| 
 | |
| def import_module(partname, fqname, parent):
 | |
|     try:
 | |
| 	return sys.modules[fqname]
 | |
|     except KeyError:
 | |
| 	pass
 | |
|     try:
 | |
| 	fp, pathname, stuff = imp.find_module(partname,
 | |
| 					      parent and parent.__path__)
 | |
|     except ImportError:
 | |
| 	return None
 | |
|     try:
 | |
| 	m = imp.load_module(fqname, fp, pathname, stuff)
 | |
|     finally:
 | |
| 	if fp: fp.close()
 | |
|     if parent:
 | |
| 	setattr(parent, partname, m)
 | |
|     return m
 | |
| 
 | |
| 
 | |
| # Replacement for reload()
 | |
| def reload_hook(module):
 | |
|     name = module.__name__
 | |
|     if '.' not in name:
 | |
| 	return import_module(name, name, None)
 | |
|     i = string.rfind(name, '.')
 | |
|     pname = name[:i]
 | |
|     parent = sys.modules[pname]
 | |
|     return import_module(name[i+1:], name, parent)
 | |
| 
 | |
| 
 | |
| # Save the original hooks
 | |
| original_import = __builtin__.__import__
 | |
| original_reload = __builtin__.reload
 | |
| 
 | |
| # Now install our hooks
 | |
| __builtin__.__import__ = import_hook
 | |
| __builtin__.reload = reload_hook
 | 
