X-Git-Url: http://dolda2000.com/gitweb/?a=blobdiff_plain;f=wrw%2Fdispatch.py;h=15ea99e8c537e74c77006d7ea052f32ebc5b859a;hb=7be9e8bb560cd4e969992759bbcb523b40e544af;hp=5a7bda0614bdd3d94ec8b3b3fe577089e5fccaa0;hpb=d7604bee761fb76faf0f70e26c946dae83ebc68e;p=wrw.git diff --git a/wrw/dispatch.py b/wrw/dispatch.py index 5a7bda0..15ea99e 100644 --- a/wrw/dispatch.py +++ b/wrw/dispatch.py @@ -1,3 +1,6 @@ +import sys, traceback +import env + __all__ = ["restart"] class restart(Exception): @@ -13,15 +16,59 @@ def mangle(result): return result return [str(result)] +class iterproxy(object): + # Makes sure iter(real).next() is called immediately, in order to + # let generator code run. + def __init__(self, real): + self.bk = real + self.bki = iter(real) + self._next = [None] + self.next() + + def __iter__(self): + return self + + def next(self): + if self._next is None: + raise StopIteration() + ret = self._next[0] + try: + self._next[:] = [self.bki.next()] + except StopIteration: + self._next = None + return ret + + def close(self): + if hasattr(self.bk, "close"): + self.bk.close() + +def defaulterror(req, excinfo): + import resp + traceback.print_exception(*excinfo) + raise resp.httperror(500) + +def wraphandler(handler, excinfo): + def wrapped(req): + return handler(req, excinfo) + return wrapped + +errorhandler = env.var(defaulterror) + def handle(req, startreq, handler): + eh = errorhandler.val try: resp = [""] while True: try: - resp = handler(req) + resp = iterproxy(handler(req)) break except restart, i: handler = i.handle + except Exception, i: + if eh is None: + raise + handler = wraphandler(eh, sys.exc_info()) + eh = None req.commit(startreq) return resp finally: