跳转至

依赖项 - 第一步

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

声明依赖项

与在路径操作函数中使用 BodyQuery 等参数的方式相同,声明依赖项使用 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 的方式与 BodyQuery 相同,但 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 以进行注册或执行类似的操作。

只要把它传递给 DependsFastAPI 就知道该如何执行后续操作。

要不要使用 async

FastAPI 调用依赖项的方式与路径操作函数一样,因此,定义依赖项函数也要使用与路径操作函数相同的规则。

即,既可以使用异步函数,也可以使用普通函数定义依赖项。

在普通路径操作函数中,可以声明异步依赖项;也可以在异步路径操作函数中声明普通依赖项。

上述这些操作都是可行的,FastAPI 知道该怎么处理。

笔记

如里不了解异步,请参阅异步:“等不及了?” 一节中 asyncawait 的内容。

与 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 文档中显示。