diff options
author | Guido van Rossum <guido@python.org> | 1995-04-07 09:03:10 (GMT) |
---|---|---|
committer | Guido van Rossum <guido@python.org> | 1995-04-07 09:03:10 (GMT) |
commit | 5232590bf144efc886a14584bd41642d3b87095f (patch) | |
tree | bde3aea7e8433d0bec4331cccf9faa926a75ddcd /Lib/newimp.py | |
parent | e9df727176faca694b5ef4daf1f423acdc683c3d (diff) | |
download | cpython-5232590bf144efc886a14584bd41642d3b87095f.zip cpython-5232590bf144efc886a14584bd41642d3b87095f.tar.gz cpython-5232590bf144efc886a14584bd41642d3b87095f.tar.bz2 |
original received from Ken
Diffstat (limited to 'Lib/newimp.py')
-rwxr-xr-x | Lib/newimp.py | 956 |
1 files changed, 956 insertions, 0 deletions
diff --git a/Lib/newimp.py b/Lib/newimp.py new file mode 100755 index 0000000..24d082d --- /dev/null +++ b/Lib/newimp.py @@ -0,0 +1,956 @@ +"""Prototype of 'import' functionality enhanced to implement packages. + +Why packages? Packages enable module nesting and sibling module +imports. 'Til now, the python module namespace was flat, which +means every module had to have a unique name, in order to not +conflict with names of other modules on the load path. Furthermore, +suites of modules could not be structurally affiliated with one +another. + +With packages, a suite of, eg, email-oriented modules can include a +module named 'mailbox', without conflicting with the, eg, 'mailbox' +module of a shared-memory suite - 'email.mailbox' vs +'shmem.mailbox'. Packages also enable modules within a suite to +load other modules within their package without having the package +name hard-coded. Similarly, package suites of modules can be loaded +as a unit, by loading the package that contains them. + +Usage: once installed (newimp.install(); newimp.revert() to revert to +the prior __import__ routine), 'import ...' and 'from ... import ...' +can be used to: + + - import modules from the search path, as before. + + - import modules from within other directory "packages" on the search + path using a '.' dot-delimited nesting syntax. The nesting is fully + recursive. + + - import an entire package as a unit, by importing the package directory. + If there is a module named '__main__.py' in the package, it controls the + load. Otherwise, all the modules in the dir, including packages, are + inherently loaded into the package module's namespace. + + __main__.py can load the entire directory, by loading the package + itself, via eg 'import __', or even 'from __ import *'. The benefit + is (1) the ability to do additional things before and after the loads + of the other modules, and (2) the ability to populate the package + module with the *contents* of the component modules, ie with a + 'from __ import *'.) + + - import siblings from modules within a package, using '__.' as a shorthand + prefix to refer to the parent package. This enables referential + transparency - package modules need not know their package name. + + - The '__' package references are actually names assigned within + modules, to refer to their containing package. This means that + variable references can be made to imported modules, or variables + defined via 'import ... from' of the modules, also using the '__.var' + shorthand notation. This establishes an proper equivalence between + the import reference '__.sibling' and the var reference '__.sibling'. + +Modules have a few new attributes, in support of packages. As +mentioned above, '__' is a shorthand attribute denoting the +modules' parent package, also denoted in the module by +'__package__'. Additionally, modules have associated with them a +'__pkgpath__', a path by which sibling modules are found.""" + +__version__ = "$Revision$" + +# $Id$ +# First release: Ken.Manheimer@nist.gov, 5-Apr-1995, for python 1.2 + +# Developers Notes: +# +# - 'sys.stub_modules' registers "incidental" (partially loaded) modules. +# A stub module is promoted to the fully-loaded 'sys.modules' list when it is +# explicitly loaded as a unit. +# - The __main__ loads of '__' have not yet been tested. +# - The test routines are cool, including a transient directory +# hierarchy facility, and a means of skipping to later tests by giving +# the test routine a numeric arg. +# - This could be substantially optimized, and there are many loose ends +# lying around, since i wanted to get this released for 1.2. + +VERBOSE = 0 + +import sys, string, regex, types, os, marshal, new, __main__ +try: + import imp # Build on this recent addition +except ImportError: + raise ImportError, 'Pkg import module depends on optional "imp" module' + +from imp import SEARCH_ERROR, PY_SOURCE, PY_COMPILED, C_EXTENSION +PY_PACKAGE = 4 # In addition to above PY_* + +modes = {SEARCH_ERROR: 'SEARCH_ERROR', + PY_SOURCE: 'PY_SOURCE', + PY_COMPILED: 'PY_COMPILED', + C_EXTENSION: 'C_EXTENSION', + PY_PACKAGE: 'PY_PACKAGE'} + +# sys.stub_modules tracks modules partially loaded modules, ie loaded only +# incidental to load of nested components. + +try: sys.stub_modules +except AttributeError: + sys.stub_modules = {} + +# Environment setup - "root" module, '__python__' + +# Establish root package '__python__' in __main__ and newimp envs. + +PKG_MAIN_NM = '__main__' # 'pkg/__main__.py' master, if present. +PKG_NM = '__package__' # Longhand for module's container. +PKG_SHORT_NM = '__' # Shorthand for module's container. +PKG_SHORT_NM_LEN = len(PKG_SHORT_NM) +PKG_PATH = '__pkgpath__' # Var holding package search path, + # usually just the path of the pkg dir. +__python__ = __main__ +sys.modules['__python__'] = __python__ # Register as an importable module. +__python__.__dict__[PKG_PATH] = sys.path + +origImportFunc = None +def install(): + """Install newimp import_module() routine, for package support. + + newimp.revert() reverts to __import__ routine that was superceded.""" + global origImportFunc + if not origImportFunc: + try: + import __builtin__ + origImportFunc = __builtin__.__import__ + except AttributeError: + pass + __builtin__.__import__ = import_module + print 'Enhanced import functionality installed.' +def revert(): + """Revert to original __builtin__.__import__ func, if newimp.install() has + been executed.""" + if origImportFunc: + import __builtin__ + __builtin__.__import__ = origImportFunc + print 'Original import routine back in place.' + +def import_module(name, + envLocals=None, envGlobals=None, + froms=None, + inPkg=None): + """Primary service routine implementing 'import' with package nesting.""" + + # The job is divided into a few distinct steps: + # + # - Look for either an already loaded module or a file to be loaded. + # * if neither loaded module nor prospect file is found, raise an error. + # - If we have a file, not an already loaded module: + # - Load the file into a module. + # - Register the new module and intermediate package stubs. + # (We have a module at this point...) + # - Bind requested syms (module or specified 'from' defs) in calling env. + # - Return the appropriate component. + + note("import_module: seeking '%s'%s" % + (name, ((inPkg and ' (in package %s)' % inPkg.__name__) or ''))) + + # We need callers environment dict for local path and resulting module + # binding. + if not (envLocals or envGlobals): + envLocals, envGlobals = exterior() + + modList = theMod = absNm = container = None + + # Get module obj if one already established, or else module file if not: + + if inPkg: + # We've been invoked with a specific containing package: + pkg, pkgPath, pkgNm = inPkg, inPkg.__dict__[PKG_PATH], inPkg.__name__ + relNm = name + absNm = pkgNm + '.' + name + + elif name[:PKG_SHORT_NM_LEN+1] != PKG_SHORT_NM + '.': + # name is NOT '__.something' - setup to seek according to specified + # absolute name. + pkg = __python__ + pkgPath = sys.path + absNm = name + relNm = absNm + + else: + # name IS '__.' + something - setup to seek according to relative name, + # in current package. + + relNm = name[len(PKG_SHORT_NM)+1:] # Relative portion of name. + try: + pkg = envGlobals[PKG_NM] # The immediately containing package. + pkgPath = pkg.__dict__[PKG_PATH] + if pkg == __python__: # At outermost package. + absNm = relNm + else: + absNm = (pkg.__name__ + '.' + relNm) + except KeyError: # Missing package, path, or name. + note("Can't identify parent package, package name, or pkgpath") + pass # ==v + + # Try to find existing module: + if sys.modules.has_key(absNm): + note('found ' + absNm + ' already imported') + theMod = sys.modules[absNm] + else: + # Try for builtin or frozen first: + theMod = imp.init_builtin(absNm) + if theMod: + note('found builtin ' + absNm) + else: + theMod = imp.init_frozen(absNm) + if theMod: + note('found frozen ' + absNm) + if not theMod: + if type(pkgPath) == types.StringType: + pkgPath = [pkgPath] + modList = find_module(relNm, pkgPath, absNm) + if not modList: + raise ImportError, "module '%s' not found" % absNm # ===X + # We have a list of successively nested files leading to the + # module, register them as stubs: + container = register_module_nesting(modList, pkg) + + # Load from file if necessary and possible: + modNm, modf, path, ty = modList[-1] + note('found type ' + modes[ty[2]] + ' - ' + absNm) + + # Do the load: + theMod = load_module(absNm, ty[2], modf, inPkg) + + # Loaded successfully - promote module to full module status: + register_module(theMod, theMod.__name__, pkgPath, pkg) + + # Have a loaded module, impose designated components, and return + # appropriate thing - according to guido: + # "Note that for "from spam.ham import bacon" your function should + # return the object denoted by 'spam.ham', while for "import + # spam.ham" it should return the object denoted by 'spam' -- the + # STORE instructions following the import statement expect it this + # way." + if not froms: + # Establish the module defs in the importing name space: + (envLocals or envGlobals)[name] = theMod + return (container or theMod) + else: + # Implement 'from': Populate immediate env with module defs: + if froms == '*': + froms = theMod.__dict__.keys() # resolve '*' + for item in froms: + (envLocals or envGlobals)[item] = theMod.__dict__[item] + return theMod + +def unload(module): + """Remove registration for a module, so import will do a fresh load.""" + if type(module) == types.ModuleType: + module = module.__name__ + for m in [sys.modules, sys.stub_modules]: + try: + del m[module] + except KeyError: + pass + +def find_module(name, path, absNm=''): + """Locate module NAME on PATH. PATH is pathname string or a list of them. + + Note that up-to-date compiled versions of a module are preferred to plain + source, and compilation is automatically performed when necessary and + possible. + + Returns a list of the tuples returned by 'find_module_file' (cf), one for + each nested level, deepest last.""" + + checked = [] # For avoiding redundant dir lists. + + if not absNm: absNm = name + + # Parse name into list of nested components, + expNm = string.splitfields(name, '.') + + for curPath in path: + + if (type(curPath) != types.StringType) or (curPath in checked): + # Disregard bogus or already investigated path elements: + continue # ==^ + else: + # Register it for subsequent disregard. + checked.append(curPath) + + if len(expNm) == 1: + + # Non-nested module name: + + got = find_module_file(curPath, absNm) + if got: + note('using %s' % got[2], 2) + return [got] # ===> + + else: + + # Composite name specifying nested module: + + gotList = []; nameAccume = expNm[0] + + got = find_module_file(curPath, nameAccume) + if not got: # Continue to next prospective path. + continue # ==^ + else: + gotList.append(got) + nm, file, fullPath, ty = got + + # Work on successively nested components: + for component in expNm[1:]: + # 'ty'pe of containing component must be package: + if ty[2] != PY_PACKAGE: + gotList, got = [], None + break # ==v^ + if nameAccume: + nameAccume = nameAccume + '.' + component + else: + nameAccume = component + got = find_module_file(fullPath, nameAccume) + if got: + gotList.append(got) + # ** have to return the *full* name here: + nm, file, fullPath, ty = got + else: + # Clear state vars: + gotList, got, nameAccume = [], None, '' + break # ==v^ + # Found nesting all the way to the specified tip: + if got: + return gotList # ===> + + # Failed. + return None + +def find_module_file(pathNm, modname): + """Find module file given dir PATHNAME and module NAME. + + If successful, returns quadruple consisting of a mod name, file object, + PATHNAME for the found file, and a description triple as contained in the + list returned by get_suffixes. + + Otherwise, returns None. + + Note that up-to-date compiled versions of a module are preferred to plain + source, and compilation is automatically performed, when necessary and + possible.""" + + relNm = string.splitfields(modname,'.')[-1] + + if pathNm[-1] != '/': pathNm = pathNm + '/' + + for suff, mode, ty in get_suffixes(): + note('trying ' + pathNm + relNm + suff + '...', 3) + fullPath = pathNm + relNm + suff + try: + modf = open(fullPath, mode) + except IOError: + # ?? Skip unreadable ones. + continue # ==^ + + if ty == PY_PACKAGE: + # Enforce directory characteristic: + if not os.path.isdir(fullPath): + note('Skipping non-dir match ' + fullPath) + continue # ==^ + else: + return (modname, modf, fullPath, (suff, mode, ty)) # ===> + + + elif ty == PY_SOURCE: + # Try for a compiled version: + note('found source ' + fullPath, 2) + pyc = fullPath + 'c' # Sadly, we're presuming '.py' suff. + if (not os.path.exists(pyc) or + (os.stat(fullPath)[8] > os.stat(pyc)[8])): + # Try to compile: + pyc = compile_source(fullPath, modf) + if pyc and (os.stat(fullPath)[8] < os.stat(pyc)[8]): + # Either pyc was already newer or we just made it so; in either + # case it's what we crave: + return (modname, open(pyc, 'rb'), pyc, # ===> + ('.pyc', 'rb', PY_COMPILED)) + # Couldn't get a compiled version - return the source: + return (modname, modf, fullPath, (suff, mode, ty)) # ===> + + elif ty == PY_COMPILED: + # Make sure it is current, trying to compile if necessary, and + # prefer source failing that: + note('found compiled ' + fullPath, 2) + py = fullPath[:-1] # Sadly again, presuming '.pyc' suff. + if not os.path.exists(py): + note('found pyc sans py: ' + fullPath) + return (modname, modf, fullPath, (suff, mode, ty)) # ===> + elif (os.stat(py)[8] > os.stat(fullPath)[8]): + note('forced to try compiling: ' + py) + pyc = compile_source(py, modf) + if pyc: + return (modname, modf, fullPath, (suff, mode, ty)) # ===> + else: + note('failed compile - must use more recent .py') + return (modname, # ===> + open(py, 'r'), py, ('.py', 'r', PY_SOURCE)) + else: + return (modname, modf, fullPath, (suff, mode, ty)) # ===> + + elif ty == C_EXTENSION: + note('found extension ' + fullPath, 2) + return (modname, modf, fullPath, (suff, mode, ty)) # ===> + + else: + raise SystemError, 'Unanticipated (new?) module type encountered' + + return None + + +def load_module(name, ty, theFile, fromMod=None): + """Load module NAME, type TYPE, from file FILE. + + Optional arg fromMod indicated the module from which the load is being done + - necessary for detecting import of __ from a package's __main__ module. + + Return the populated module object.""" + + # Note: we mint and register intermediate package directories, as necessary + + # Determine packagepath extension: + + # Establish the module object in question: + theMod = procure_module(name) + nameTail = string.splitfields(name, '.')[-1] + thePath = theFile.name + + if ty == PY_SOURCE: + exec_into(theFile, theMod, theFile.name) + + elif ty == PY_COMPILED: + pyc = open(theFile.name, 'rb').read() + if pyc[0:4] != imp.get_magic(): + raise ImportError, 'bad magic number: ' + theFile.name # ===> + code = marshal.loads(pyc[8:]) + exec_into(code, theMod, theFile.name) + + elif ty == C_EXTENSION: + try: + theMod = imp.load_dynamic(nameTail, thePath, theFile) + except: + # ?? Ok to embellish the error message? + raise sys.exc_type, ('%s (from %s)' % + (str(sys.exc_value), theFile.name)) + + elif ty == PY_PACKAGE: + # Load constituents: + if (os.path.exists(thePath + '/' + PKG_MAIN_NM) and + # pkg has a __main__, and this import not already from __main__, so + # __main__ can 'import __', or even better, 'from __ import *' + ((theMod.__name__ != PKG_MAIN_NM) and (fromMod.__ == theMod))): + exec_into(thePath + '/' + PKG_MAIN_NM, theMod, theFile.name) + else: + # ... or else recursively load constituent modules. + prospects = mod_prospects(thePath) + for item in prospects: + theMod.__dict__[item] = import_module(item, + theMod.__dict__, + theMod.__dict__, + None, + theMod) + + else: + raise ImportError, 'Unimplemented import type: %s' % ty # ===> + + return theMod + +def exec_into(obj, module, path): + """Helper for load_module, execfile/exec path or code OBJ within MODULE.""" + + # This depends on ability of exec and execfile to mutilate, erhm, mutate + # the __dict__ of a module. It will not work if/when this becomes + # disallowed, as it is for normal assignments. + + try: + if type(obj) == types.FileType: + execfile(path, module.__dict__, module.__dict__) + elif type(obj) in [types.CodeType, types.StringType]: + exec obj in module.__dict__, module.__dict__ + except: + # ?? Ok to embellish the error message? + raise sys.exc_type, ('%s (from %s)' % + (str(sys.exc_value), path)) + + +def mod_prospects(path): + """Return a list of prospective modules within directory PATH. + + We actually return the distinct names resulting from stripping the dir + entries (excluding '.' and '..') of their suffixes (as represented by + 'get_suffixes'). + + (Note that matches for the PY_PACKAGE type with null suffix are + implicitly constrained to be directories.)""" + + # We actually strip the longest matching suffixes, so eg 'dbmmodule.so' + # mates with 'module.so' rather than '.so'. + + dirList = os.listdir(path) + excludes = ['.', '..'] + sortedSuffs = sorted_suffixes() + entries = [] + for item in dirList: + if item in excludes: continue # ==^ + for suff in sortedSuffs: + sub = -1 * len(suff) + if sub == 0: + if os.path.isdir(os.path.join(path, item)): + entries.append(item) + elif item[sub:] == suff: + it = item[:sub] + if not it in entries: + entries.append(it) + break # ==v^ + return entries + + + +def procure_module(name): + """Return an established or else new module object having NAME. + + First checks sys.modules, then sys.stub_modules.""" + + if sys.modules.has_key(name): + it = sys.modules[name] + elif sys.stub_modules.has_key(name): + it = sys.stub_modules[name] + else: + it = new.module(name) + return it # ===> + +def register_module_nesting(modList, pkg): + """Given a find_module()-style NESTING and a parent PACKAGE, register + components as stub modules.""" + container = None + for stubModNm, stubModF, stubPath, stubTy in modList: + relStubNm = string.splitfields(stubModNm, '.')[-1] + if sys.modules.has_key(stubModNm): + # Nestle in containing package: + stubMod = sys.modules[stubModNm] + pkg.__dict__[relStubNm] = stubMod + pkg = stubMod # will be parent for next in sequence. + elif sys.stub_modules.has_key(stubModNm): + stubMod = sys.stub_modules[stubModNm] + pkg.__dict__[relStubNm] = stubMod + pkg = stubMod + else: + stubMod = procure_module(stubModNm) + # Register as a stub: + register_module(stubMod, stubModNm, stubPath, pkg, 1) + pkg.__dict__[relStubNm] = stubMod + pkg = stubMod + if not container: + container = stubMod + return container + +def register_module(theMod, name, path, package, stub=0): + """Properly register MODULE, w/ name, path, package, opt, as stub.""" + + if stub: + sys.stub_modules[name] = theMod + else: + sys.modules[name] = theMod + if sys.stub_modules.has_key(name): + del sys.stub_modules[name] + theMod.__ = theMod.__dict__[PKG_NM] = package + theMod.__dict__[PKG_PATH] = path + + +def compile_source(sourcePath, sourceFile): + """Given python code source path and file obj, Create a compiled version. + + Return path of compiled version, or None if file creation is not + successful. (Compilation errors themselves are passed without restraint.) + + This is an import-private interface, and not well-behaved for general use. + + In particular, we presume the validity of the sourcePath, and that it + includes a '.py' extension.""" + + compiledPath = sourcePath[:-3] + '.pyc' + try: + compiledFile = open(compiledPath, 'wb') + except IOError: + note("write permission denied to " + compiledPath) + return None + mtime = os.stat(sourcePath)[8] + sourceFile.seek(0) # rewind + try: + compiledFile.write(imp.get_magic()) # compiled magic number + compiledFile.seek(8, 0) # mtime space holder + # We let compilation errors go their own way... + compiled = compile(sourceFile.read(), sourcePath, 'exec') + marshal.dump(compiled, compiledFile) # write the code obj + compiledFile.seek(4, 0) # position for mtime + compiledFile.write(marshal.dumps(mtime)[1:]) # register mtime + compiledFile.flush() + compiledFile.close() + return compiledPath + except IOError: + return None + + +def PathExtension(locals, globals): # Probably obsolete. + """Determine import search path extension vis-a-vis __pkgpath__ entries. + + local dict __pkgpath__ will preceed global dict __pkgpath__.""" + + pathadd = [] + if globals and globals.has_key(PKG_PATH): + pathadd = PrependPath(pathadd, globals[PKG_PATH], 'global') + if locals and locals.has_key(PKG_PATH): + pathadd = PrependPath(pathadd, locals[PKG_PATH], 'local') + if pathadd: + note(PKG_PATH + ' extension: ' + pathadd) + return pathadd + +def PrependPath(path, pre, whence): # Probably obsolete + """Return copy of PATH list with string or list-of-strings PRE prepended. + + If PRE is neither a string nor list-of-strings, print warning that + locality WHENCE has malformed value.""" + + # (There is probably a better way to handle malformed PREs, but raising an + # error seems too severe - in that case, a bad setting for + # sys.__pkgpath__ would prevent any imports!) + + if type(pre) == types.StringType: return [pre] + path[:] # ===> + elif type(pre) == types.ListType: return pre + path[:] # ===> + else: + print "**Ignoring '%s' bad %s value**" % (whence, PKG_PATH) + return path # ===> + +got_suffixes = None +def get_suffixes(): + """Produce a list of triples, each describing a type of import file. + + Triples have the form '(SUFFIX, MODE, TYPE)', where: + + SUFFIX is a string found appended to a module name to make a filename for + that type of import file. + + MODE is the mode string to be passed to the built-in 'open' function - "r" + for text files, "rb" for binary. + + TYPE is the file type: + + PY_SOURCE: python source code, + PY_COMPILED: byte-compiled python source, + C_EXTENSION: compiled-code object file, + PY_PACKAGE: python library directory, or + SEARCH_ERROR: no module found. """ + + # Note: sorted_suffixes() depends on this function's value being invariant. + # sorted_suffixes() must be revised if this becomes untrue. + + global got_suffixes + + if got_suffixes: + return got_suffixes + else: + # Ensure that the .pyc suffix precedes the .py: + got_suffixes = [('', 'r', PY_PACKAGE)] + py = pyc = None + for suff in imp.get_suffixes(): + if suff[0] == '.py': + py = suff + elif suff[0] == '.pyc': + pyc = suff + else: + got_suffixes.append(suff) + got_suffixes.append(pyc) + got_suffixes.append(py) + return got_suffixes + + +sortedSuffs = [] # State vars for sorted_suffixes(). Go +def sorted_suffixes(): + """Helper function ~efficiently~ tracks sorted list of module suffixes.""" + + # Produce sortedSuffs once - this presumes that get_suffixes does not + # change from call to call during a python session. Needs to be + # corrected if that becomes no longer true. + + global sortedsuffs + if not sortedSuffs: # do compute only the "first" time + for item in get_suffixes(): + sortedSuffs.append(item[0]) + # Sort them in descending order: + sortedSuffs.sort(lambda x, y: (((len(x) > len(y)) and 1) or + ((len(x) < len(y)) and -1))) + sortedSuffs.reverse() + return sortedSuffs + + +# exterior(): Utility routine, obtain local and global dicts of environment +# containing/outside the callers environment, ie that of the +# caller's caller. Routines can use exterior() to determine the +# environment from which they were called. + +def exterior(): + """Return dyad containing locals and globals of caller's caller. + + Locals will be None if same as globals, ie env is global env.""" + + bogus = 'bogus' # A locally usable exception + try: raise bogus # Force an exception object + except bogus: + at = sys.exc_traceback.tb_frame.f_back # The external frame. + if at.f_back: at = at.f_back # And further, if any. + globals, locals = at.f_globals, at.f_locals + if locals == globals: # Exterior is global? + locals = None + return (locals, globals) + +######################################################################### +# TESTING FACILITIES # + +def note(msg, threshold=1): + if VERBOSE >= threshold: print '(import:', msg, ')' + +class TestDirHier: + """Populate a transient directory hierarchy according to a definition + template - so we can create package/module hierarchies with which to + exercise the new import facilities...""" + + def __init__(self, template, where='/var/tmp'): + """Establish a dir hierarchy, according to TEMPLATE, that will be + deleted upon deletion of this object (or deliberate invocation of the + __del__ method).""" + self.PKG_NM = 'tdh_' + rev = 0 + while os.path.exists(os.path.join(where, self.PKG_NM+str(rev))): + rev = rev + 1 + sys.exc_traceback = None # Ensure Discard of try/except obj ref + self.PKG_NM = self.PKG_NM + str(rev) + self.root = os.path.join(where, self.PKG_NM) + self.createDir(self.root) + self.add(template) + + def __del__(self): + """Cleanup the test hierarchy.""" + self.remove() + def add(self, template, root=None): + """Populate directory according to template dictionary. + + Keys indicate file names, possibly directories themselves. + + String values dictate contents of flat files. + + Dictionary values dictate recursively embedded dictionary templates.""" + if root == None: root = self.root + for key, val in template.items(): + name = os.path.join(root, key) + if type(val) == types.StringType: # flat file + self.createFile(name, val) + elif type(val) == types.DictionaryType: # embedded dir + self.createDir(name) + self.add(val, name) + else: + raise ValueError, 'invalid file-value type, %s' % type(val) + def remove(self, name=''): + """Dispose of the NAME (or keys in dictionary), using 'rm -r'.""" + name = os.path.join(self.root, name) + sys.exc_traceback = None # Ensure Discard of try/except obj ref + if os.path.exists(name): + print '(TestDirHier: deleting %s)' % name + os.system('rm -r ' + name) + else: + raise IOError, "can't remove non-existent " + name + def createFile(self, name, contents=None): + """Establish file NAME with CONTENTS. + + If no contents specfied, contents will be 'print NAME'.""" + f = open(name, 'w') + if not contents: + f.write("print '" + name + "'\n") + else: + f.write(contents) + f.close + def createDir(self, name): + """Create dir with NAME.""" + return os.mkdir(name, 0755) + +skipToTest = 0 +atTest = 1 +def testExec(msg, execList, locals, globals): + global skipToTest, atTest + print 'Import Test:', '(' + str(atTest) + ')', msg, '...' + atTest = atTest + 1 + if skipToTest > (atTest - 1): + print ' ... skipping til test', skipToTest + return + else: + print '' + for stmt in execList: + exec stmt in locals, globals + +def test(number=0): + """Exercise import functionality, creating a transient dir hierarchy for + the purpose. + + We actually install the new import functionality, temporarily, resuming the + existing function on cleanup.""" + + import __builtin__ + + global skipToTest, atTest + skipToTest = number + hier = None + + def confPkgVars(mod, locals, globals): + if not sys.modules.has_key(mod): + print 'import test: missing module "%s"' % mod + else: + modMod = sys.modules[mod] + if not modMod.__dict__.has_key(PKG_SHORT_NM): + print ('import test: module "%s" missing %s pkg shorthand' % + (mod, PKG_SHORT_NM)) + if not modMod.__dict__.has_key(PKG_PATH): + print ('import test: module "%s" missing %s package path' % + (mod, PKG_PATH)) + def unloadFull(mod): + """Unload module and offspring submodules, if any.""" + modMod = '' + if type(mod) == types.StringType: + modNm = mod + elif type(mod) == types.ModuleType: + modNm = modMod.__name__ + for subj in sys.modules.keys() + sys.stub_modules.keys(): + if subj[0:len(modNm)] == modNm: + unload(subj) + + # First, get the globals and locals to pass to our testExec(): + exec 'import ' + __name__ + globals, locals = eval(__name__ + '.__dict__'), vars() + + try: + __main__.testMods + except AttributeError: + __main__.testMods = [] + testMods = __main__.testMods + + + # Install the newimp routines, within a try/finally: + try: + sys.exc_traceback = None + wasImport = __builtin__.__import__ # Stash default + wasPath = sys.path + except AttributeError: + wasImport = None + try: + hiers = []; modules = [] + global VERBOSE + wasVerbose, VERBOSE = VERBOSE, 2 + __builtin__.__import__ = import_module # Install new version + + if testMods: # Clear out imports from previous tests + for m in testMods[:]: + unloadFull(m) + testMods.remove(m) + + testExec("already imported module: %s" % sys.modules.keys()[0], + ['import ' + sys.modules.keys()[0]], + locals, globals) + try: + no_sirree = 'no_sirree_does_not_exist' + testExec("non-existent module: %s" % no_sirree, + ['import ' + no_sirree], + locals, globals) + except ImportError: + testExec("ok", ['pass'], locals, globals) + got = None + for mod in ['Complex', 'UserDict', 'UserList', 'calendar', + 'cmd', 'dis', 'mailbox', 'profile', 'random', 'rfc822']: + if not (mod in sys.modules.keys()): + got = mod + break # ==v + if got: + testExec("not-yet loaded module: %s" % mod, + ['import ' + mod, 'modules.append(got)'], + locals, globals) + else: + print "Import Test: couldn't find unimported module from list" + + # Now some package stuff. + + # First change the path to include our temp dir, copying so the + # addition can be revoked on cleanup in the finally, below: + sys.path = ['/var/tmp'] + sys.path[:] + # Now create a trivial package: + stmts = ["hier1 = TestDirHier({'a.py': 'print \"a.py executing\"'})", + "hiers.append(hier1)", + "root = hier1.PKG_NM", + "exec 'import ' + root", + "testMods.append(root)", + "confPkgVars(sys.modules[root].__name__, locals, globals)", + "confPkgVars(sys.modules[root].__name__+'.a',locals,globals)"] + testExec("trivial package, with one module, a.py", + stmts, locals, globals) + # Slightly less trivial package - reference to '__': + stmts = [("hier2 = TestDirHier({'ref.py': 'print \"Pkg __:\", __'})"), + "root = hier2.PKG_NM", + "hiers.append(hier2)", + "exec 'import ' + root", + "testMods.append(root)"] + testExec("trivial package, with module that has pkg shorthand ref", + stmts, locals, globals) + # Nested package, plus '__' references: + + complexTemplate = {'ref.py': 'print "ref.py loading..."', + 'suite': {'s1.py': 'print "s1.py, in pkg:", __', + 'subsuite': {'sub1.py': + 'print "sub1.py"'}}} + stmts = [('print """%s\n%s\n%s\n%s\n%s\n%s"""' % + ('.../', + ' ref.py\t\t\t"ref.py loading..."', + ' suite/', + ' s1.py \t\t"s1.py, in pkg: xxxx.suite"', + ' subsuite/', + ' sub1.py "sub1.py" ')), + "hier3 = TestDirHier(complexTemplate)", + "root = hier3.PKG_NM", + "hiers.append(hier3)", + "exec 'import ' + root", + "testMods.append(root)"] + testExec("Significantly nestled package:", + stmts, locals, globals) + + # Now try to do an embedded sibling import, using '__' shorthand - + # alter our complexTemplate for a new dirHier: + complexTemplate['suite']['s1.py'] = 'import __.subsuite' + stmts = ["hier4 = TestDirHier(complexTemplate)", + "root = hier4.PKG_NM", + "testMods.append(root)", + "hiers.append(hier4)", + "exec 'import %s.suite.s1' % root", + "testMods.append(root)"] + testExec("Similar structure, but suite/s1.py imports '__.subsuite'", + stmts, locals, globals) + + sys.exc_traceback = None # Signify clean conclusion. + + finally: + if sys.exc_traceback: + print ' ** Import test FAILURE... cleanup.' + else: + print ' Import test SUCCESS... cleanup' + VERBOSE = wasVerbose + skipToTest = 0 + atTest = 1 + sys.path = wasPath + for h in hiers: h.remove(); del h # Dispose of test directories + if wasImport: # Resurrect prior routine + __builtin__.__import__ = wasImport + else: + del __builtin__.__import__ |