首页 游戏 软件 资讯 排行榜 专题
首页
编程语言
Python pytest怎么对FastAPI进行异步测试_使用httpx与pytest-asyncio

Python pytest怎么对FastAPI进行异步测试_使用httpx与pytest-asyncio

热心网友
64
转载
2026-05-06

Python pytest怎么对FastAPI进行异步测试_使用httpx与pytest-asyncio

Python pytest怎么对FastAPI进行异步测试_使用httpx与pytest-asyncio

免费影视、动漫、音乐、游戏、小说资源长期稳定更新! 👉 点此立即查看 👈

为什么不能直接用 requests 测试 FastAPI 的 async endpoint

这里有个常见的误区:直接用 requests 库去测试 FastAPI 的异步端点。问题出在哪儿?关键在于,requests 是一个同步阻塞的库,而 FastAPI 中那些用 async def 定义的路由函数,其内部逻辑是异步的。如果你直接调用 requests.get(“https://localhost:8000/”),很可能会触发一个运行时错误,典型的提示是:RuntimeError: There is no current event loop in thread ‘MainThread’。这背后的核心原因在于,同步请求无法正确“等待”(await)路由内部的协程逻辑,比如一个 await database.fetch_one(...) 操作。即便你手动启动了事件循环,测试结果也往往是不可靠的,无法真实反映异步接口的行为。

用 httpx.AsyncClient + pytest-asyncio 是最轻量可靠的组合

那么,正确的姿势是什么?答案是 httpx.AsyncClientpytest-asyncio 的组合。这套组合拳之所以被推崇,原因很直接:httpx.AsyncClient 原生支持异步 HTTP 调用,能够真实模拟客户端的行为;而 pytest-asyncio 提供的 @pytest.mark.asyncio 修饰器,可以自动管理事件循环的生命周期,省去了手动调用 asyncio.run()loop.run_until_complete() 的麻烦。两者配合,代码既干净,调试也友好,还能完美兼容 pytest 强大的 fixture 机制。

首先,安装必要的依赖:

pip install httpx pytest-asyncio

为了省去在每个测试函数上都写 @pytest.mark.asyncio 的麻烦,可以在 conftest.py 文件或测试文件的顶部添加如下配置:

立即学习“Python免费学习笔记(深入)”;

pytest_plugins = [“pytest_asyncio”]

一个基础的测试示例如下:

import pytest
from httpx import AsyncClient
from myapp.main import app  # 假设你的 FastAPI 实例叫 app

@pytest.mark.asyncio
async def test_read_root():
    async with AsyncClient(app=app, base_url=“https://www.php.cn/link/9688c999c6508777280b6e8074ad82fa”) as ac:
        response = await ac.get(“/”)
        assert response.status_code == 200
        assert response.json() == {“hello”: “world”}
  • 参数 app=app 表示直接使用 ASGI 模式连接应用,无需启动真实的服务器,测试速度快且环境可控。
  • base_url 参数必须设置(哪怕只是一个占位符),否则调用 ac.get(“/foo”) 时会因为缺少主机信息而报 MissingSchema 错误。
  • 务必使用 async with 语句来管理客户端,否则客户端不会被正确清理,可能引发警告甚至资源泄漏。

如何在测试中复用 FastAPI 的依赖(比如数据库 session)

FastAPI 中通过 Depends() 注入的依赖,在测试环境中并不会自动生效。这就需要我们显式地进行覆盖。常见的做法是利用 pytest 的 fixture 机制,将生产环境的依赖(如数据库连接)替换为测试专用的版本,例如内存 SQLite 或测试专用的 Session。

假设你的路由函数是这样的:def read_items(db: Session = Depends(get_db))。在测试时,可以这样覆盖依赖:

from myapp.dependencies import get_db
from sqlalchemy.ext.asyncio import create_async_engine, AsyncSession

@pytest.fixture(scope=“function”)
def test_session():
    engine = create_async_engine(“sqlite+aiosqlite:///:memory:”, echo=False)
    AsyncSessionLocal = sessionmaker(engine, class_=AsyncSession, expire_on_commit=False)
    return AsyncSessionLocal()

@pytest.mark.asyncio
async def test_read_items(test_session):
    app.dependency_overrides[get_db] = lambda: test_session
    async with AsyncClient(app=app, base_url=“https://www.php.cn/link/9688c999c6508777280b6e8074ad82fa”) as ac:
        response = await ac.get(“/items/”)
        assert response.status_code == 200
    app.dependency_overrides.clear()  # ⚠️ 必须清理,否则会污染其他测试
  • app.dependency_overrides 是一个字典,赋值后必须手动调用 .clear() 进行清理,pytest 不会自动重置它。
  • 如果被覆盖的依赖本身是异步函数(比如 async def get_db()),那么你覆盖的 lambda 函数也需要返回一个可等待对象(awaitable),或者直接使用 lambda: get_db()(注意:这会返回一个协程,需要在依赖解析时被 await)。
  • 内存 SQLite 可能不支持某些特定数据库(如 PostgreSQL)的特性(例如 RETURNING 子句),在涉及复杂 SQL 的场景下,建议使用 pytest-async-sqlalchemy 这类工具,或者连接真实的测试数据库。

遇到 “Event loop closed” 或 “Task was destroyed but it is pending” 怎么办

这类错误在异步测试中并不少见,通常是因为在 fixture 中创建了未被正确等待(await)的异步任务,或者客户端、会话等资源没有被正确关闭。最常见的两个“坑”是:

  • 在 fixture 中使用了 asyncio.create_task(...) 但没有后续的 await tasktask.cancel(),导致任务在事件循环关闭前仍然挂起。
  • 多个测试用例共享了一个未做隔离的全局异步资源(例如一个单例数据库连接池),当一个测试结束时,连接池可能还在处理请求。

解决办法其实很直接:

  • 为所有异步 fixture 都加上 scope=“function” 参数,确保每个测试函数都使用独立的资源实例,避免跨测试复用。
  • 尽量避免使用 create_task,改用 await 直接调用协程;如果确实需要并发,使用 asyncio.gather(...) 并确保所有任务都被等待完成。
  • 仔细检查 AsyncClient 是否都在 async with 代码块内使用——遗漏会导致底层的 httpcore.AsyncConnectionPool 未能正常关闭。

真正让调试变得棘手的地方在于,这些错误可能不是每次测试都会出现,尤其是在 CI 环境中,它们往往是偶发的。所以,一个黄金法则是:只要测试涉及异步 fixture 或后台任务,一律按照“每个测试独占其资源”的原则来设计,这样可以最大程度避免不可预知的竞态条件和资源冲突。

来源:https://www.php.cn/faq/2314637.html
免责声明: 游乐网为非赢利性网站,所展示的游戏/软件/文章内容均来自于互联网或第三方用户上传分享,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系youleyoucom@outlook.com。

相关攻略

Python怎么将多个特征处理步骤组合_FeatureUnion合并多种提取器
编程语言
Python怎么将多个特征处理步骤组合_FeatureUnion合并多种提取器

Python怎么将多个特征处理步骤组合_FeatureUnion合并多种提取器 FeatureUnion 在 scikit-learn 中早已被弃用 先说一个明确的结论:FeatureUnion 这个工具,从 scikit-learn 1 2 版本开始就被官方标记为弃用(deprecated)了。如

热心网友
05.06
Python如何监听全局键盘按键实现自动化快捷键触发
编程语言
Python如何监听全局键盘按键实现自动化快捷键触发

Python如何监听全局键盘按键实现自动化快捷键触发 你是否希望在Python中设置一个全局快捷键?例如,无论你当前正在编辑文档、浏览网页还是运行游戏,只需按下Ctrl+Shift+X这样的组合键,就能自动执行预设的自动化任务。这个需求听起来直观,但在实际开发中,会面临跨平台兼容性、系统权限以及逻辑

热心网友
05.06
Python如何统计分组内不重复的元素个数_聚合时指定nunique统计函数
编程语言
Python如何统计分组内不重复的元素个数_聚合时指定nunique统计函数

Python分组去重计数:掌握nunique()函数,提升数据分析效率 在数据分析工作中,按组统计唯一值数量是一项常见且关键的任务。例如,分析每个产品类别下的独立访客数,或计算每个销售区域每年上架的不同商品种类。此时,pandas库中的nunique()函数便成为高效解决此类问题的首选工具。 nun

热心网友
05.06
Python自动化识别验证码图片_tesseract-ocr实现OCR识别
编程语言
Python自动化识别验证码图片_tesseract-ocr实现OCR识别

Tesseract OCR 识别失败的核心原因在于输入图像质量不佳且缺乏针对性预处理。必须进行二值化、形态学去噪、倾斜校正等操作,并配合使用 --psm 8 参数和字符白名单;通过 Python 调用时需显式传递配置参数,在 Windows 系统上还需指定 tesseract_cmd 路径;调试过程

热心网友
05.06
Python怎么销毁一个对象_探究__del__析构函数与垃圾回收机制
编程语言
Python怎么销毁一个对象_探究__del__析构函数与垃圾回收机制

Python对象销毁机制详解:__del__析构函数与垃圾回收的正确使用 Python中__del__方法的局限性:为何它不是可靠的销毁钩子 需要明确的是,Python的__del__方法**无法保证一定会被执行**,因此不适合用于释放文件句柄、网络连接或数据库事务等关键系统资源。它仅仅是CPyth

热心网友
05.06

最新APP

宝宝过生日
宝宝过生日
应用辅助 04-07
台球世界
台球世界
体育竞技 04-07
解绳子
解绳子
休闲益智 04-07
骑兵冲突
骑兵冲突
棋牌策略 04-07
三国真龙传
三国真龙传
角色扮演 04-07

热门推荐

Composer生成vendor离线包详细步骤与实用指南
编程语言
Composer生成vendor离线包详细步骤与实用指南

vendor目录离线包本质是composer install --no-dev后的完整快照 vendor 目录离线包本质是 composer install --no-dev 后的完整快照 Composer vendor目录离线包,本质上是一个经过精简、可直接部署到生产环境的依赖文件夹快照。其核心目

热心网友
05.06
CentOS系统设置PHP定时任务详细步骤
编程语言
CentOS系统设置PHP定时任务详细步骤

在CentOS系统中设置PHP定时任务 对于需要在CentOS服务器上自动化执行PHP脚本的场景,crontab无疑是那个最经典、最可靠的工具。它就像一位不知疲倦的守夜人,能帮你精准地按计划完成任务。下面,我们就来一步步拆解如何配置它。 第一步:确保PHP环境就绪 首先,需要确认您的CentOS系统

热心网友
05.06
CentOS系统安装PHP依赖的详细步骤
编程语言
CentOS系统安装PHP依赖的详细步骤

在CentOS上安装PHP依赖的完整指南 想要在CentOS系统中高效部署PHP扩展?首要步骤并非直接执行安装指令,而是配置好功能强大的“软件源仓库”。EPEL与Remi仓库是构建稳定PHP环境的基石。本教程将详细解析从仓库配置到扩展安装的全流程,助你搭建坚实的PHP运行基础。 安装EPEL仓库 E

热心网友
05.06
CentOS系统配置PHP远程数据库连接教程
编程语言
CentOS系统配置PHP远程数据库连接教程

CentOS系统下PHP远程连接配置指南:基于cURL扩展的完整教程 在CentOS服务器环境中,实现PHP与外部网络资源的远程通信是常见的开发需求。cURL扩展作为PHP内置的强大网络库,能够高效支持HTTP、HTTPS、FTP等多种协议的数据传输。本教程将详细演示如何在CentOS系统上配置并使

热心网友
05.06
CentOS系统下配置vsFTPd服务集成指南
编程语言
CentOS系统下配置vsFTPd服务集成指南

在CentOS上集成vsftpd与其他服务:一份实战指南 将CentOS系统中的vsftpd(Very Secure FTP Daemon)与其他关键服务进行集成,能够大幅增强其功能性、安全性与管理效率。具体的集成方案需根据您的实际业务需求来定制。本文将深入探讨几个最常见的集成场景,并提供清晰、可操

热心网友
05.06