Flask请求钩子与上下文对象
-
请求钩子
在客户端和服务器交互的过程中,有些准备工作或扫尾工作需要处理,比如:
- 在请求开始时,建立数据库连接;
- 在请求开始时,根据需求进行权限校验;
- 在请求结束时,指定数据的交互格式;
为了让每个视图函数避免编写重复功能的代码,Flask提供了通用设施的功能,即请求钩子。
请求钩子是通过装饰器的形式实现,Flask支持如下四种请求钩子:
-
before_first_request
- 在处理第一个请求前执行
-
before_request
- 在每次请求前执行
- 如果在某修饰的函数中返回了一个响应,视图函数将不再被调用
-
after_request
- 如果没有抛出错误,在每次请求后执行
- 接受一个参数:视图函数作出的响应
- 在此函数中可以对响应值在返回之前做最后一步修改处理
- 需要将参数中的响应在此参数中进行返回
-
teardown_request:
- 在after_request请求后执行
- 接受一个参数:错误信息,如果有相关错误抛出
-
代码实例
from flask import Flask app = Flask(__name__) # 请求钩子 # 在处理第一个请求前执行 @app.before_first_request def before_first_request(): print("before_first_request") # 在每一次请求之前调用 @app.before_request def before_request(): print("before_request") # 在执行完视图函数之后会调用,并且会把视图函数所生成的响应传入 @app.after_request def after_request(response): print("after_request") response.headers["Content-Type"] = "application/json" return response # 请每一次请求之后都会调用,会接受一个参数,参数是服务器出现的错误信息 @app.teardown_request def teardown_request(e): print("teardown_request") @app.route('/') def index(): return 'Hello World' if __name__ == '__main__': app.run()
-
上下文对象
-
上下文
上下文:即语境,语意,在程序中可以理解为在代码执行到某一时刻时,根据之前代码所做的操作以及下文即将要执行的逻辑,可以决定在当前时刻下可以使用到的变量,或者可以完成的事情。
Flask中有两种上下文,请求上下文和应用上下文
Flask中上下文对象:相当于一个容器,保存了 Flask 程序运行过程中的一些信息。
-
请求上下文(request context)
-
场景
在视图函数中,如何取到当前请求的相关数据?比如:请求地址,请求方式,cookie等等
-
解决:
在 flask 中,可以直接在视图函数中使用 request 这个对象进行获取相关数据,而 request 就是请求上下文的对象,保存了当前本次请求的相关数据,请求上下文对象有:request、session
- request
- 封装了HTTP请求的内容,针对的是http请求。
- session
- 用来记录请求会话中的信息,针对的是用户信息。
- request
-
-
应用上下文(application context)
-
介绍
它的字面意思是应用上下文,但它不是一直存在的,它只是request context 中的一个对 app 的代理(人),所谓local proxy。它的作用主要是帮助 request 获取当前的应用,它是伴 request 而生,随 request 而灭的。应用上下文对象有:current_app,g
-
current_app对象
-
介绍
应用程序上下文,用于存储应用程序中的变量,可以通过current_app.name打印当前app的名称,也可以在current_app中存储一些变量,例如:
- 应用的启动脚本是哪个文件,启动时指定了哪些参数
- 加载了哪些配置文件,导入了哪些配置
- 连了哪个数据库
- 有哪些public的工具类、常量
- 应用跑再哪个机器上,IP多少,内存多大
-
示例
09flask_context中
from flask import Flask, current_app app = Flask(__name__) app.secret_key = 'TPmi4aLWRbyVq8zu9v82dWYW1' class Config(object): REDIS = "redis://127.0.0.1/1" app.config.from_object(Config) @app.route('/') def index(): return 'Hello World' from flask02.user import user_bp app.register_blueprint(user_bp, url_prefix="/user") if __name__ == '__main__': app.run()
user模块代码
from flask import Blueprint, current_app user_bp = Blueprint("user", __name__) # current_app获取配置信息 @user_bp.route("/") def user(): # redis://127.0.0.1/1 print(current_app.config["REDIS"]) # 09flask_context print(current_app.name) # 所有的路由信息 print(current_app.url_map) return "user模块" @user_bp.route("/info") def user_info(): return "user_info"
-
作用
current_app 就是当前运行的flask app,在代码不方便直接操作flask的app对象时,可以操作current_app就等价于操作flask app对象。
-
-
g对象
-
介绍
g 作为 flask 程序全局的一个临时变量,充当中间媒介的作用,我们可以通过它在一次请求调用的多个函数间传递一些数据。每次请求都会重设这个变量。
-
示例
from flask import Flask, g app = Flask(__name__) def db_query(): user_id = g.user_id user_name = g.user_name print('user_id={} user_name={}'.format(user_id, user_name)) @app.route('/') def get_user_profile(): g.user_id = 123 g.user_name = 'xlz' db_query() return 'hello world'
-
-
g对象与请求钩子的综合案例
-
需求
- 构建认证机制
- 对于特定视图可以提供强制要求用户登录的限制
- 对于所有视图,无论是否强制要求用户登录,都可以在视图中尝试获取用户认证后的身份信息
-
实现
from flask import Flask, session, g, abort app = Flask(__name__) app.secret_key = 'TPmi4aLWRbyVq8zu9v82dWYW1' @app.before_request def authentication(): g.name = session.get("name") def login_required(func): def wrapper(*args, **kwargs): if g.name is not None: return func(*args, **kwargs) else: # abort(401) return "请先登录" return wrapper @app.route('/login') def index(): session["name"] = "xlz" return '登录用户' @app.route('/user') @login_required def get_user(): return '你好,用户{}'.format(g.name) if __name__ == '__main__': app.run()
-
-
app_context 与 request_context
-
场景
在Flask程序未运行的情况下,调试代码时需要使用
current_app
、g
、request
这些对象,会不会有问题?该如何使用? -
app_context
app_context
为我们提供了应用上下文环境,允许我们在外部使用应用上下文current_app
、g
,可以通过
with
语句进行使用。from flask import Flask, current_app, g app = Flask(__name__) app.secret_key = 'TPmi4aLWRbyVq8zu9v82dWYW1' @app.route('/') def index(): return 'Hello World' if __name__ == '__main__': # 构造一个应用上下文,在应用上下文中,curent_app和g可用, request和session不可用 with app.app_context(): g.name = "xlz" print(current_app.secret_key) print(g.name) app.run()
-
request_context
request_context
为我们提供了请求上下文环境,允许我们在外部使用请求上下文request
、session
,可以通过with语句进行使用。from flask import Flask, request, session, current_app, g app = Flask(__name__) @app.route('/') def index(): return 'Hello World' if __name__ == '__main__': # 模拟解析客户端请求之后的wsgi字典数据 environ = { 'wsgi.version': (1, 0), 'wsgi.input': '', 'REQUEST_METHOD': 'GET', 'PATH_INFO': '/', 'SERVER_NAME': 'itcast server', 'wsgi.url_scheme': 'http', 'SERVER_PORT': '80' } with app.request_context(environ): print(request.path) print(session.get("name")) print(current_app) print(g) app.run()
-
test_request_context
test_request_context允许我们在外部使用4个上下文对象,可以通过
with
语句进行使用from flask import Flask, request, session, current_app, g app = Flask(__name__) @app.route('/') def index(): return 'Hello World' if __name__ == '__main__': # 在请求外面,使用请求上下文对象,报错 # 如要使用可以构造一个测试的请求上下文,4个上下文对象都可用 with app.test_request_context('/'): print(request.url) print(session.get("name")) print(current_app) print(g) app.run()
-
-
-