forked from orbit-oss/flask
Added a workaround for a pypy bug in context managers
This commit is contained in:
parent
de7e2bcf7a
commit
6ec83e18dc
2 changed files with 31 additions and 0 deletions
|
|
@ -71,3 +71,27 @@ def with_metaclass(meta, *bases):
|
||||||
return type.__new__(cls, name, (), d)
|
return type.__new__(cls, name, (), d)
|
||||||
return meta(name, bases, d)
|
return meta(name, bases, d)
|
||||||
return metaclass('temporary_class', None, {})
|
return metaclass('temporary_class', None, {})
|
||||||
|
|
||||||
|
|
||||||
|
# Certain versions of pypy have a bug where clearing the exception stack
|
||||||
|
# breaks the __exit__ function in a very peculiar way. This is currently
|
||||||
|
# true for pypy 2.2.1 for instance. The second level of exception blocks
|
||||||
|
# is necessary because pypy seems to forget to check if an exception
|
||||||
|
# happend until the next bytecode instruction?
|
||||||
|
BROKEN_PYPY_CTXMGR_EXIT = False
|
||||||
|
if hasattr(sys, 'pypy_version_info'):
|
||||||
|
class _Mgr(object):
|
||||||
|
def __enter__(self):
|
||||||
|
return self
|
||||||
|
def __exit__(self, *args):
|
||||||
|
sys.exc_clear()
|
||||||
|
try:
|
||||||
|
try:
|
||||||
|
with _Mgr():
|
||||||
|
raise AssertionError()
|
||||||
|
except:
|
||||||
|
raise
|
||||||
|
except TypeError:
|
||||||
|
BROKEN_PYPY_CTXMGR_EXIT = True
|
||||||
|
except AssertionError:
|
||||||
|
pass
|
||||||
|
|
|
||||||
|
|
@ -19,6 +19,7 @@ from werkzeug.exceptions import HTTPException
|
||||||
from .globals import _request_ctx_stack, _app_ctx_stack
|
from .globals import _request_ctx_stack, _app_ctx_stack
|
||||||
from .module import blueprint_is_module
|
from .module import blueprint_is_module
|
||||||
from .signals import appcontext_pushed, appcontext_popped
|
from .signals import appcontext_pushed, appcontext_popped
|
||||||
|
from ._compat import BROKEN_PYPY_CTXMGR_EXIT
|
||||||
|
|
||||||
|
|
||||||
class _AppCtxGlobals(object):
|
class _AppCtxGlobals(object):
|
||||||
|
|
@ -187,6 +188,9 @@ class AppContext(object):
|
||||||
def __exit__(self, exc_type, exc_value, tb):
|
def __exit__(self, exc_type, exc_value, tb):
|
||||||
self.pop(exc_value)
|
self.pop(exc_value)
|
||||||
|
|
||||||
|
if BROKEN_PYPY_CTXMGR_EXIT and exc_type is not None:
|
||||||
|
raise exc_type, exc_value, tb
|
||||||
|
|
||||||
|
|
||||||
class RequestContext(object):
|
class RequestContext(object):
|
||||||
"""The request context contains all request relevant information. It is
|
"""The request context contains all request relevant information. It is
|
||||||
|
|
@ -390,6 +394,9 @@ class RequestContext(object):
|
||||||
# See flask.testing for how this works.
|
# See flask.testing for how this works.
|
||||||
self.auto_pop(exc_value)
|
self.auto_pop(exc_value)
|
||||||
|
|
||||||
|
if BROKEN_PYPY_CTXMGR_EXIT and exc_type is not None:
|
||||||
|
raise exc_type, exc_value, tb
|
||||||
|
|
||||||
def __repr__(self):
|
def __repr__(self):
|
||||||
return '<%s \'%s\' [%s] of %s>' % (
|
return '<%s \'%s\' [%s] of %s>' % (
|
||||||
self.__class__.__name__,
|
self.__class__.__name__,
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue