83 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			83 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Python
		
	
	
	
| import py
 | |
| defaultconftestpath = py.path.local(__file__).dirpath("defaultconftest.py")
 | |
| 
 | |
| class Conftest(object):
 | |
|     """ the single place for accessing values and interacting 
 | |
|         towards conftest modules from py.test objects. 
 | |
| 
 | |
|         Note that triggering Conftest instances to import 
 | |
|         conftest.py files may result in added cmdline options. 
 | |
|         XXX
 | |
|     """ 
 | |
|     def __init__(self, path=None, onimport=None):
 | |
|         self._path2confmods = {}
 | |
|         self._onimport = onimport
 | |
|         if path is not None:
 | |
|             self.setinitial([path])
 | |
| 
 | |
|     def setinitial(self, args):
 | |
|         """ try to find a first anchor path for looking up global values
 | |
|             from conftests. This function is usually called _before_  
 | |
|             argument parsing.  conftest files may add command line options
 | |
|             and we thus have no completely safe way of determining
 | |
|             which parts of the arguments are actually related to options
 | |
|             and which are file system paths.  We just try here to get 
 | |
|             bootstrapped ... 
 | |
|         """
 | |
|         current = py.path.local()
 | |
|         for arg in args + [current]:
 | |
|             anchor = current.join(arg, abs=1)
 | |
|             if anchor.check(): # we found some file object 
 | |
|                 self._path2confmods[None] = self.getconftestmodules(anchor)
 | |
|                 break
 | |
|         else:
 | |
|             assert 0, "no root of filesystem?"
 | |
| 
 | |
|     def getconftestmodules(self, path):
 | |
|         """ return a list of imported conftest modules for the given path.  """ 
 | |
|         try:
 | |
|             clist = self._path2confmods[path]
 | |
|         except KeyError:
 | |
|             if path is None:
 | |
|                 raise ValueError("missing default conftest.")
 | |
|             dp = path.dirpath()
 | |
|             if dp == path: 
 | |
|                 return [self.importconftest(defaultconftestpath)]
 | |
|             clist = self.getconftestmodules(dp)
 | |
|             conftestpath = path.join("conftest.py")
 | |
|             if conftestpath.check(file=1):
 | |
|                 clist.append(self.importconftest(conftestpath))
 | |
|             self._path2confmods[path] = clist
 | |
|         # be defensive: avoid changes from caller side to
 | |
|         # affect us by always returning a copy of the actual list 
 | |
|         return clist[:]
 | |
| 
 | |
|     def rget(self, name, path=None):
 | |
|         mod, value = self.rget_with_confmod(name, path)
 | |
|         return value
 | |
| 
 | |
|     def rget_with_confmod(self, name, path=None):
 | |
|         modules = self.getconftestmodules(path)
 | |
|         modules.reverse()
 | |
|         for mod in modules:
 | |
|             try:
 | |
|                 return mod, getattr(mod, name)
 | |
|             except AttributeError:
 | |
|                 continue
 | |
|         raise KeyError(name)
 | |
| 
 | |
|     def importconftest(self, conftestpath):
 | |
|         # Using caching here looks redundant since ultimately
 | |
|         # sys.modules caches already 
 | |
|         assert conftestpath.check(), conftestpath
 | |
|         if not conftestpath.dirpath('__init__.py').check(file=1): 
 | |
|             # HACK: we don't want any "globally" imported conftest.py, 
 | |
|             #       prone to conflicts and subtle problems 
 | |
|             modname = str(conftestpath).replace('.', conftestpath.sep)
 | |
|             mod = conftestpath.pyimport(modname=modname)
 | |
|         else:
 | |
|             mod = conftestpath.pyimport()
 | |
|         if self._onimport:
 | |
|             self._onimport(mod)
 | |
|         return mod
 |