Flask框架中信号基于blinker,可以让开发人员在flask请求过程中定制一些用户行为
- 安装blinker:pip3 install blinker
- blinker提供的信号
request_started = _signals.signal(‘request-started‘) # 请求到来前执行 request_finished = _signals.signal(‘request-finished‘) # 请求结束后执行 before_render_template = _signals.signal(‘before-render-template‘) # 模板渲染前执行 template_rendered = _signals.signal(‘template-rendered‘) # 模板渲染后执行 got_request_exception = _signals.signal(‘got-request-exception‘) # 请求执行出现异常时执行 request_tearing_down = _signals.signal(‘request-tearing-down‘) # 请求执行完毕后自动执行(无论成功与否) appcontext_tearing_down = _signals.signal(‘appcontext-tearing-down‘)# 请求上下文执行完毕后自动执行(无论成功与否) appcontext_pushed = _signals.signal(‘appcontext-pushed‘) # 请求上下文push时执行 appcontext_popped = _signals.signal(‘appcontext-popped‘) # 请求上下文pop时执行 message_flashed = _signals.signal(‘message-flashed‘) # 调用flask在其中添加数据时,自动触发
- 源码:
class Flask(_PackageBoundObject): def wsgi_app(self, environ, start_response) ctx = self.request_context(environ) ctx.push() error = None try: try: response = self.full_dispatch_request() #调用full_dispatch_request方法 except Exception as e: error = e response = self.handle_exception(e) except: error=sys.exc_info()[1] raise return response(environ, start_response) finally: if self.should_ignore_error(error): error = None ctx.auto_pop(error)
def full_dispatch_request(self): self.try_trigger_before_first_request_functions() #执行@before_first_request这个装饰器装饰的函数 try:##################触发request_started信号############################
request_started.send(self) rv = self.preprocess_request() #执行@before_request这个装饰器装饰的函数 if rv is None: rv = self.dispatch_request()
except Exception as e: rv = self.handle_user_exception(e) return self.finalize_request(rv)
def finalize_request(self, rv, from_error_handler=False): response = self.make_response(rv) try: response = self.process_response(response) #执行@after_request这个装饰器装饰的函数同时会保存session ###################触发request_finished信号###############
request_finished.send(self, response=response) except Exception: if not from_error_handler: raise self.logger.exception(‘Request finalizing failed with an ‘ ‘error while handling an error‘) return response
def _render(template, context, app): ####################触发before_render_template信号############### before_render_template.send(app, template=template, context=context) rv = template.render(context) ###################触发templated_rendered信号##################### template_rendered.send(app, template=template, context=context) return rv def render_template(template_name_or_list, **context): ctx = _app_ctx_stack.top ctx.app.update_template_context(context) return _render(ctx.app.jinja_env.get_or_select_template(template_name_or_list), context, ctx.app)
class Flask(_PackageBoundObject): def wsgi_app(self, environ, start_response) ctx = self.request_context(environ) ctx.push() error = None try: try: response = self.full_dispatch_request() except Exception as e: error = e response = self.handle_exception(e) #调用handle_exception方法 except: error=sys.exc_info()[1] raise return response(environ, start_response) finally: if self.should_ignore_error(error): error = None ctx.auto_pop(error)
def handle_exception(self, e): exc_type, exc_value, tb = sys.exc_info() #################触发错误处理信号############# got_request_exception.send(self, exception=e) handler = self._find_error_handler(InternalServerError()) if self.propagate_exceptions: if exc_value is e: reraise(exc_type, exc_value, tb) else: raise e self.log_exception((exc_type, exc_value, tb)) if handler is None: return InternalServerError() return self.finalize_request(handler(e), from_error_handler=True)
class AppContext(object): def push(self): self._refcnt += 1 if hasattr(sys, ‘exc_clear‘): sys.exc_clear() _app_ctx_stack.push(self) ######触发appcontext_pushed信号######### appcontext_pushed.send(self.app) def pop(self, exc=_sentinel): """Pops the app context.""" try: self._refcnt -= 1 if self._refcnt <= 0: if exc is _sentinel: exc = sys.exc_info()[1] #调用Flask类中的do_teardown_appcontext方法,触发appcontext_tearing_down信号 self.app.do_teardown_appcontext(exc) finally: rv = _app_ctx_stack.pop() assert rv is self, ‘Popped wrong app context. (%r instead of %r)‘ % (rv, self) #######触发appcontext_popped信号####### appcontext_popped.send(self.app) class RequestContext(object): def pop(self, exc=_sentinel): app_ctx = self._implicit_app_ctx_stack.pop() try: clear_request = False if not self._implicit_app_ctx_stack: self.preserved = False self._preserved_exc = None if exc is _sentinel: exc = sys.exc_info()[1] #调用Flask类中调用do_teardown_request方法,触发 request_tearing_down信号 self.app.do_teardown_request(exc) if hasattr(sys, ‘exc_clear‘): sys.exc_clear() request_close = getattr(self.request, ‘close‘, None) if request_close is not None: request_close() clear_request = True finally: rv = _request_ctx_stack.pop() if clear_request: rv.request.environ[‘werkzeug.request‘] = None if app_ctx is not None: app_ctx.pop(exc) assert rv is self, ‘Popped wrong request context. ‘ ‘(%r instead of %r)‘ % (rv, self) class Flask(_PackageBoundObject): def do_teardown_request(self, exc=_sentinel): if exc is _sentinel: exc = sys.exc_info()[1] funcs = reversed(self.teardown_request_funcs.get(None, ())) bp = _request_ctx_stack.top.request.blueprint if bp is not None and bp in self.teardown_request_funcs: funcs = chain(funcs, reversed(self.teardown_request_funcs[bp])) for func in funcs: func(exc) #########触发request_tearing_down信号######## request_tearing_down.send(self, exc=exc) def do_teardown_appcontext(self, exc=_sentinel): if exc is _sentinel: exc = sys.exc_info()[1] for func in reversed(self.teardown_appcontext_funcs): func(exc) ##########触发appcontext_tearing_down信号 appcontext_tearing_down.send(self, exc=exc)
自定义信号
一个信号需满足三个条件:有信号、信号中必须注册函数、请求触发函数 from flask import Flask,current_app,flash,render_template from flask.signals import _signals app=Flask(__name__) # 自定义信号 sig=_signals.signal(‘sig‘) def func(sender,*args,**kwargs): print(sender) # 给自定义信号中注册函数 sig.connect(func) @app.route(‘/‘) def hello_world(): # 触发信号 sig.send(‘hello world‘) return ‘hello world‘ if __name__ == ‘__main__‘: app.run()
原文地址:https://www.cnblogs.com/meng0410/p/8659357.html
时间: 2024-10-22 08:41:09