依赖项 - 第一步¶
FastAPI 提供了简单易用,但功能强大的依赖注入系统。
这个依赖系统设计的简单易用,开发者可以轻松地把各种组件集成至 FastAPI。
什么是依赖注入¶
编程中的依赖注入是声明代码(本文中为路径操作函数 )运行所需的,或要使用的依赖的一种方式。
然后,由系统(本文中为 FastAPI)负责执行任意需要的逻辑,为代码提供这些依赖(注入依赖项)。
依赖注入常用于以下场景:
- 共享业务逻辑(复用相同的代码逻辑)
- 共享数据库连接
- 实现安全、验证、角色权限
- 等等……
上述场景均可以使用依赖注入,将代码重复最小化。
第一步¶
接下来,我们学习一个非常简单的例子,尽管它过于简单,不是很实用。
但通过这个例子,您可以初步了解依赖注入的工作机制。
创建依赖项¶
首先,要关注的是依赖项。
依赖项是使用与路径操作函数相同参数的函数:
from typing import Union
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(
q: Union[str, None] = None, skip: int = 0, limit: int = 100
):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
@app.get("/users/")
async def read_users(commons: dict = Depends(common_parameters)):
return commons
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(q: str | None = None, skip: int = 0, limit: int = 100):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
@app.get("/users/")
async def read_users(commons: dict = Depends(common_parameters)):
return commons
搞定收工。
只用了2 行代码。
依赖项函数的形式和结构与路径操作函数一样。
因此,依赖项就相当于没有装饰器(即,没有 @app.get("/some-path")
)的路径操作函数。
依赖项可以返回各种内容。
本例的依赖项接收如下参数:
str
类型的可选查询参数q
int
类型的可选查询参数skip
,默认值是0
int
类型的可选查询参数limit
,默认值是100
依赖项函数返回的是包含这些值的字典。
导入 Depends
¶
from typing import Union
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(
q: Union[str, None] = None, skip: int = 0, limit: int = 100
):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
@app.get("/users/")
async def read_users(commons: dict = Depends(common_parameters)):
return commons
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(q: str | None = None, skip: int = 0, limit: int = 100):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
@app.get("/users/")
async def read_users(commons: dict = Depends(common_parameters)):
return commons
声明依赖项¶
与在路径操作函数中使用 Body
、Query
等参数的方式相同,声明依赖项使用 Depends
和一个新的参数:
from typing import Union
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(
q: Union[str, None] = None, skip: int = 0, limit: int = 100
):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
@app.get("/users/")
async def read_users(commons: dict = Depends(common_parameters)):
return commons
from fastapi import Depends, FastAPI
app = FastAPI()
async def common_parameters(q: str | None = None, skip: int = 0, limit: int = 100):
return {"q": q, "skip": skip, "limit": limit}
@app.get("/items/")
async def read_items(commons: dict = Depends(common_parameters)):
return commons
@app.get("/users/")
async def read_users(commons: dict = Depends(common_parameters)):
return commons
虽然,在路径操作函数的参数中使用 Depends
的方式与 Body
、Query
相同,但 Depends
的工作方式略有不同。
这里只能传给 Depends 一个参数。
且该参数必须是可调用对象,比如函数。
该函数接收的参数和路径操作函数的参数一样。
提示
下一节介绍,除了函数之外,依赖项还可以使用哪些对象。
接收到新的请求时,FastAPI 执行如下操作:
- 用正确的参数调用依赖项函数(可依赖项)
- 获取函数返回的结果
- 把函数返回的结果赋值给路径操作函数的参数
graph TB
common_parameters(["common_parameters"])
read_items["/items/"]
read_users["/users/"]
common_parameters --> read_items
common_parameters --> read_users
这样,只编写一次代码,FastAPI 就可以为多个路径操作共享这段代码 。
检查
注意,无需创建专门的类,并将之传递给 FastAPI 以进行注册或执行类似的操作。
只要把它传递给 Depends
,FastAPI 就知道该如何执行后续操作。
要不要使用 async
?¶
FastAPI 调用依赖项的方式与路径操作函数一样,因此,定义依赖项函数也要使用与路径操作函数相同的规则。
即,既可以使用异步函数,也可以使用普通函数定义依赖项。
在普通路径操作函数中,可以声明异步依赖项;也可以在异步路径操作函数中声明普通依赖项。
上述这些操作都是可行的,FastAPI 知道该怎么处理。
笔记
如里不了解异步,请参阅异步:“等不及了?” 一节中 async
和 await
的内容。
与 OpenAPI 集成¶
依赖项及子依赖项的所有请求声明、验证和需求都可以集成至同一个 OpenAPI 概图。
所以,交互文档里也会显示依赖项的所有信息:
简单用法¶
观察一下就会发现,只要路径和操作匹配,就可以使用声明的路径操作函数。然后,FastAPI 会用正确的参数调用函数,并提取请求中的数据。
实际上,所有(或大多数)网络框架的工作方式都是这样的。
开发者永远都不需要直接调用这些函数,这些函数是由框架(在此为 FastAPI )调用的。
通过依赖注入系统,只要告诉 FastAPI 路径操作函数 还要依赖其他在路径操作函数之前执行的内容,FastAPI 就会执行函数代码,并注入函数返回的结果。
其他与依赖注入概念相同的术语为:
- 资源(Resource)
- 提供方(Provider)
- 服务(Service)
- 可注入对象(Injectable)
- 组件(Component)
FastAPI 插件¶
依赖注入系统支持插件。但实际上,FastAPI 并不需要创建插件,因为使用依赖项可以声明不限数量的、可用于路径操作函数的集成与交互。
创建依赖项非常简单、直观,而且还支持导入 Python 包。毫不夸张地说,只要几行代码就可以把 Python 包与 API 函数集成在一起。
下一节详细介绍如何关系型数据库、NoSQL 数据库、安全措施中使用依赖项。
FastAPI 兼容性¶
依赖注入系统如此简洁的特性,让 FastAPI 与下列系统兼容:
- 关系型数据库
- NoSQL 数据库
- 外部支持库
- 外部 API
- 验证和鉴权系统
- API 使用监控系统
- 响应数据注入系统
- 等等……
简单而强大¶
虽然,层级式依赖注入系统的定义与使用十分简单,但它却非常强大。
例如,可以定义依赖其他依赖项的依赖项。
最后,依赖项层级树构建后,依赖注入系统会处理所有依赖项及其子依赖项,并为每一步操作提供(注入)结果。
例如,下面有 4 个 API 端点(路径操作):
/items/public/
/items/private/
/users/{user_id}/activate
/items/pro/
开发者使用依赖项及其子依赖项就能为以下路径操作添加不同权限:
graph TB
current_user(["current_user"])
active_user(["active_user"])
admin_user(["admin_user"])
paying_user(["paying_user"])
public["/items/public/"]
private["/items/private/"]
activate_user["/users/{user_id}/activate"]
pro_items["/items/pro/"]
current_user --> active_user
active_user --> admin_user
active_user --> paying_user
current_user --> public
active_user --> private
admin_user --> activate_user
paying_user --> pro_items
与 OpenAPI 集成¶
声明需求时,依赖项还会把参数、验证等功能添加至路径操作。
FastAPI 负责把它们全部添加到 OpenAPI 概图,并在 API 文档中显示。