FastAPI 中间件

中间件是一个函数,它在请求特定路径操作之前,以及在每个响应之后执行。

使用装饰器创建中间件

创建一个函数,在函数的顶部使用装饰器 @app.middleware("http").

函数参数如下:

  • request: Request

  • call_next 函数:这个函数将接收 request 作为参数,将 request 传递给相应的 路径操作,然后它将返回由相应的路径操作生成的 response.

python 复制代码
import uvicorn
from fastapi import FastAPI, Request

app = FastAPI()

@app.middleware("http")
async def m2(request: Request, call_next):
    """第二个中间件"""
    print("第二个中间件 m2 ...request")

    response = await call_next(request)

    print("第二个中间件 m2 ...response")

    return response

@app.middleware("http")
async def m1(request: Request, call_next):
    """第一个中间件"""
    print("第一个中间件 m1 ...request")

    response = await call_next(request)

    print("第一个中间件 m1 ...response")

    return response

@app.get("/item/{item_id}")
async def item(item_id: int):
    print("请求get item id", item_id)
    return {"item_id": item_id}

if __name__ == '__main__':
    uvicorn.run(app)

运行结果:

python 复制代码
第一个中间件 m1 ...request
第二个中间件 m2 ...request
请求get item id 11111
第二个中间件 m2 ...response
第一个中间件 m1 ...response
INFO:     127.0.0.1:58664 - "GET /item/11111 HTTP/1.1" 200 OK

继承BaseHTTPMiddleware

1、定义中间件类:首先,你需要定义一个类,并继承自 starlette.middleware.base.BaseHTTPMiddleware。

2、实现 dispatch 方法:在你的类中,你需要实现 dispatch 方法。这个方法会在请求被传递给下一个中间件或最终的应用程序之前被调用。

3、注册你的中间件:在你的 FastAPI 应用中注册你的中间件。

创建my_middlewares包

示例:计算API接口调用耗时 中间件

创建CalculateTimeMiddleware.py 文件

python 复制代码
import time

from fastapi import Request, FastAPI
from starlette.middleware.base import BaseHTTPMiddleware

class CalculateTimeMiddleware(BaseHTTPMiddleware):
    """
    计算API接口耗时 单位: 秒
    """

    async def dispatch(self, request: Request, call_next):
        print("中间件: 计算API接口耗时...")
        start_time = time.perf_counter()
        response = await call_next(request)
        process_time = time.perf_counter() - start_time
        print(f"请求路径: {request.url.path}, 耗时: {process_time} 秒")
        return response

编写Main.py 文件

python 复制代码
import uvicorn
from fastapi import FastAPI
from starlette.requests import Request

from my_middlewares.AuthMiddleware import AuthMiddleware
from my_middlewares.CalculateTimeMiddleware import CalculateTimeMiddleware

app = FastAPI()

# 注册中间件
app.add_middleware(AuthMiddleware)  # type: ignore
app.add_middleware(CalculateTimeMiddleware) # type: ignore

@app.get("/item/{item_id}")
async def item(item_id: int, request: Request):
	# 获取上下文变量
    user_id = request.state.user_id
    print("user_id:", user_id)

    return {"item_id": item_id}

if __name__ == '__main__':
    uvicorn.run(app)

示例:中间件传递参数

使用上下文变量(Context Variables),FastAPI的Request对象允许你设置和获取上下文变量,这在多个中间件或路由处理器之间共享数据非常有用。

编写AuthMiddleware.py文件

python 复制代码
from starlette.middleware.base import BaseHTTPMiddleware

class AuthMiddleware(BaseHTTPMiddleware):
    """
    权限校验,并传递 user_id
    """

    async def dispatch(self, request, call_next):
        print("中间件: 传递 user_id ...")
        user_id="user001"
	
        # 设置上下文变量 将user_id存储在request.state中
        request.state.user_id = user_id

        response = await call_next(request)

        return response

运行结果:

python 复制代码
中间件: 计算API接口耗时...
中间件: 传递 user_id ...
user_id: user001
请求路径: /item/111, 耗时: 0.0006680000005871989 秒
相关推荐
Ares-Wang9 小时前
FastAPI 数据验证 Pydantic Flask 用 WTForms
python·flask·fastapi
曲幽10 小时前
FastAPI自动生成的API文档太丑?我花了一晚上把它改成了客户愿意付费的样子
python·fastapi·web·swagger·openapi·scalar·docs
PieroPc1 天前
一个功能强大的 Web 端标签设计和打印工具,支持服务器端直接打印到局域网打印机。Fastapi + html
前端·html·fastapi
别抢我的锅包肉2 天前
FastAPI + Vue3 + Vite 跨域报错全解:从 `Access-Control-Allow-Origin missing` 到彻底修复
中间件·状态模式·fastapi
Chase_______2 天前
【FastAPI】内网/离线环境docs文档无法显示的解决方案
fastapi
小李云雾2 天前
FastAPI 后端开发:文件上传 + 表单提交
开发语言·python·lua·postman·fastapi
曲幽2 天前
告别手写 API 胶水代码:FastAPI 与 Vue 的“契约自动机” OpenAPI 实战
python·typescript·vue·fastapi·web·swagger·openapi·codegen
2401_835792543 天前
FastAPI 速通
windows·python·fastapi
海市公约3 天前
FastAPI入门实战:从零搭建到核心功能详解
fastapi·后端入门·异步编程·路由管理·python web开发·api设计·pydantic数据验证
曲幽4 天前
FastAPI + Vue 前后端分离实战:我的项目结构“避坑指南”
python·vue·fastapi·web·vite·proxy·cors·env