Browse Source

Merge pull request #1822 from pallets/bugfix/better-pop

Improve application context popping
pull/1823/head
Armin Ronacher 9 years ago
parent
commit
8d7e7aab31
  1. 2
      CHANGES
  2. 80
      flask/ctx.py
  3. 22
      tests/test_appctx.py

2
CHANGES

@ -77,6 +77,8 @@ Version 0.11
- ``send_from_directory`` now raises BadRequest if the filename is invalid on - ``send_from_directory`` now raises BadRequest if the filename is invalid on
the server OS (pull request ``#1763``). the server OS (pull request ``#1763``).
- Added the ``JSONIFY_MIMETYPE`` configuration variable (pull request ``#1728``). - Added the ``JSONIFY_MIMETYPE`` configuration variable (pull request ``#1728``).
- Exceptions during teardown handling will no longer leave bad application
contexts lingering around.
Version 0.10.2 Version 0.10.2
-------------- --------------

80
flask/ctx.py

@ -181,12 +181,14 @@ class AppContext(object):
def pop(self, exc=_sentinel): def pop(self, exc=_sentinel):
"""Pops the app context.""" """Pops the app context."""
self._refcnt -= 1 try:
if self._refcnt <= 0: self._refcnt -= 1
if exc is _sentinel: if self._refcnt <= 0:
exc = sys.exc_info()[1] if exc is _sentinel:
self.app.do_teardown_appcontext(exc) exc = sys.exc_info()[1]
rv = _app_ctx_stack.pop() self.app.do_teardown_appcontext(exc)
finally:
rv = _app_ctx_stack.pop()
assert rv is self, 'Popped wrong app context. (%r instead of %r)' \ assert rv is self, 'Popped wrong app context. (%r instead of %r)' \
% (rv, self) % (rv, self)
appcontext_popped.send(self.app) appcontext_popped.send(self.app)
@ -341,38 +343,40 @@ class RequestContext(object):
""" """
app_ctx = self._implicit_app_ctx_stack.pop() app_ctx = self._implicit_app_ctx_stack.pop()
clear_request = False try:
if not self._implicit_app_ctx_stack: clear_request = False
self.preserved = False if not self._implicit_app_ctx_stack:
self._preserved_exc = None self.preserved = False
if exc is _sentinel: self._preserved_exc = None
exc = sys.exc_info()[1] if exc is _sentinel:
self.app.do_teardown_request(exc) exc = sys.exc_info()[1]
self.app.do_teardown_request(exc)
# If this interpreter supports clearing the exception information
# we do that now. This will only go into effect on Python 2.x, # If this interpreter supports clearing the exception information
# on 3.x it disappears automatically at the end of the exception # we do that now. This will only go into effect on Python 2.x,
# stack. # on 3.x it disappears automatically at the end of the exception
if hasattr(sys, 'exc_clear'): # stack.
sys.exc_clear() if hasattr(sys, 'exc_clear'):
sys.exc_clear()
request_close = getattr(self.request, 'close', None)
if request_close is not None: request_close = getattr(self.request, 'close', None)
request_close() if request_close is not None:
clear_request = True request_close()
clear_request = True
rv = _request_ctx_stack.pop() finally:
assert rv is self, 'Popped wrong request context. (%r instead of %r)' \ rv = _request_ctx_stack.pop()
% (rv, self)
# get rid of circular dependencies at the end of the request
# get rid of circular dependencies at the end of the request # so that we don't require the GC to be active.
# so that we don't require the GC to be active. if clear_request:
if clear_request: rv.request.environ['werkzeug.request'] = None
rv.request.environ['werkzeug.request'] = None
# Get rid of the app as well if necessary.
# Get rid of the app as well if necessary. if app_ctx is not None:
if app_ctx is not None: app_ctx.pop(exc)
app_ctx.pop(exc)
assert rv is self, 'Popped wrong request context. ' \
'(%r instead of %r)' % (rv, self)
def auto_pop(self, exc): def auto_pop(self, exc):
if self.request.environ.get('flask._preserve_context') or \ if self.request.environ.get('flask._preserve_context') or \

22
tests/test_appctx.py

@ -146,3 +146,25 @@ def test_context_refcounts():
assert res.status_code == 200 assert res.status_code == 200
assert res.data == b'' assert res.data == b''
assert called == ['request', 'app'] assert called == ['request', 'app']
def test_clean_pop():
called = []
app = flask.Flask(__name__)
@app.teardown_request
def teardown_req(error=None):
1 / 0
@app.teardown_appcontext
def teardown_app(error=None):
called.append('TEARDOWN')
try:
with app.test_request_context():
called.append(flask.current_app.name)
except ZeroDivisionError:
pass
assert called == ['test_appctx', 'TEARDOWN']
assert not flask.current_app

Loading…
Cancel
Save