[docs]classErrorDuringImport(Exception):"""Errors that occurred while trying to import something"""def__init__(self,filename,exc_info):self.filename=filenameself.exc,self.value,self.tb=exc_infodef__str__(self):exc=self.exc.__name__return"problem in %s - %s: %s"%(self.filename,exc,self.value)
[docs]defsafeimport(path,forceload=0,cache=None):"""Import a module; handle errors; return None if the module isn't found. If the module *is* found but an exception occurs, it's wrapped in an ErrorDuringImport exception and reraised. Unlike __import__, if a package path is specified, the module at the end of the path is returned, not the package at the beginning. If the optional 'forceload' argument is 1, we reload the module from disk (unless it's a dynamic extension). """ifcacheisNone:cache=dict()LOG.debug("Trying to import '%s'"%path)try:# If forceload is 1 and the module has been previously loaded from# disk, we always have to reload the module. Checking the file's# mtime isn't good enough (e.g. the module could contain a class# that inherits from another module that has changed).ifforceloadandpathinsys.modules:ifpathnotinsys.builtin_module_names:# Remove the module from sys.modules and re-import to try# and avoid problems with partially loaded modules.# Also remove any submodules because they won't appear# in the newly loaded module's namespace if they're# already in sys.modules.subs=[mforminsys.modulesifm.startswith(path+".")]forkeyin[path]+subs:# Prevent garbage collection.cache[key]=sys.modules[key]delsys.modules[key]module=__import__(path)exceptException:# Did the error occur before or after the module was found?(exc,value,tb)=info=sys.exc_info()ifpathinsys.modules:# An error occurred while executing the imported module.raiseErrorDuringImport(sys.modules[path].__file__,info)elifexcisSyntaxError:# A SyntaxError occurred before we could execute the module.raiseErrorDuringImport(value.filename,info)elifexcisImportErrorandvalue.name==path:# No such module in the path.returnNoneelse:# Some other error occurred during the importing process.raiseErrorDuringImport(path,sys.exc_info())forpartinpath.split(".")[1:]:try:module=getattr(module,part)exceptAttributeError:returnNonereturnmodule
[docs]deflocate(path,forceload=0):"""Locate an object by name or dotted path, importing as necessary."""parts=[partforpartinpath.split(".")ifpart]module,n=None,0whilen<len(parts)-1:nextmodule=safeimport(".".join(parts[:n+1]),forceload)ifnextmodule:module,n=nextmodule,n+1else:breakifmodule:object=moduleelse:object=builtinsforpartinparts[n:]:try:object=getattr(object,part)exceptAttributeErroraserr:LOG.error("Could not load '%s': %s"%(part,err))returnNonereturnobject
[docs]defload_with_params(module_path,params):"""Load a class from *module_path* and pass *params*."""module,clazz=module_path.split(":")module=import_module(module)returngetattr(module,clazz)(**params)