""" py lib plugins and plugin call management """ import py class MultiCall: """ Manage a specific call into many python functions/methods. Simple example: MultiCall([list1.append, list2.append], 42).execute() """ NONEASRESULT = object() def __init__(self, methods, *args, **kwargs): self.methods = methods[:] self.args = args self.kwargs = kwargs self.results = [] def execute(self, firstresult=False): while self.methods: currentmethod = self.methods.pop() res = self.execute_method(currentmethod) if hasattr(self, '_ex1'): self.results = [res] break if res is not None: if res is self.NONEASRESULT: res = None self.results.append(res) if firstresult: break if not firstresult: return self.results if self.results: return self.results[-1] def execute_method(self, currentmethod): self.currentmethod = currentmethod # provide call introspection if "__call__" is the first positional argument if hasattr(currentmethod, 'im_self'): varnames = currentmethod.im_func.func_code.co_varnames needscall = varnames[1:2] == ('__call__',) else: try: varnames = currentmethod.func_code.co_varnames except AttributeError: # builtin function varnames = () needscall = varnames[:1] == ('__call__',) if needscall: return currentmethod(self, *self.args, **self.kwargs) else: #try: return currentmethod(*self.args, **self.kwargs) #except TypeError: # print currentmethod.__module__, currentmethod.__name__, self.args, self.kwargs # raise def exclude_other_results(self): self._ex1 = True class Registry: """ Manage Plugins: Load plugins and manage calls to plugins. """ MultiCall = MultiCall def __init__(self, plugins=None): if plugins is None: plugins = [] self.plugins = plugins def register(self, plugin): assert not isinstance(plugin, str) self.call_each("pytest_plugin_registered", plugin) self.plugins.append(plugin) def unregister(self, plugin): self.call_each("pytest_plugin_unregistered", plugin) self.plugins.remove(plugin) def isregistered(self, plugin): return plugin in self.plugins def __iter__(self): return iter(self.plugins) def listattr(self, attrname, plugins=None, extra=(), reverse=False): l = [] if plugins is None: plugins = self.plugins if extra: plugins += list(extra) for plugin in plugins: try: l.append(getattr(plugin, attrname)) except AttributeError: continue if reverse: l.reverse() return l def call_each(self, methname, *args, **kwargs): """ return call object for executing a plugin call. """ return MultiCall(self.listattr(methname), *args, **kwargs).execute() def call_firstresult(self, methname, *args, **kwargs): """ return first non-None result of a plugin method. """ return MultiCall(self.listattr(methname), *args, **kwargs).execute(firstresult=True) def call_plugin(self, plugin, methname, *args, **kwargs): return MultiCall(self.listattr(methname, plugins=[plugin]), *args, **kwargs).execute(firstresult=True) class PluginAPI: def __init__(self, apiclass, plugins=None): self._apiclass = apiclass if plugins is None: plugins = comregistry self.plugins = plugins for name, method in vars(apiclass).items(): if name[:2] != "__": firstresult = getattr(method, 'firstresult', False) mm = ApiCall(plugins, name, firstresult=firstresult) setattr(self, name, mm) def __repr__(self): return "" %(self._apiclass, self.plugins) class ApiCall: def __init__(self, plugins, name, firstresult): self.plugins = plugins self.name = name self.firstresult = firstresult def __repr__(self): mode = self.firstresult and "firstresult" or "each" return "" %(self.name, mode, self.plugins) def __call__(self, *args, **kwargs): mc = MultiCall(self.plugins.listattr(self.name), *args, **kwargs) #print "making multicall", self return mc.execute(firstresult=self.firstresult) comregistry = Registry()