news 2026/5/5 13:59:36

故障排查:Pytest Asyncio Event Loop Closed 错误

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
故障排查:Pytest Asyncio Event Loop Closed 错误

1. 问题描述

在运行RetrievalService的集成测试(使用pytest-asyncio)时,当连续运行多个异步测试用例时,遇到了以下错误:

RuntimeError: Task <Task pending ...> got Future <Future pending ...> attached to a different loop ... RuntimeError: Event loop is closed

症状

  • 第一个测试用例 (test_search_knowledge_base_flow) 成功通过。
  • 第二个测试用例 (test_search_knowledge_base_no_results) 在 setup 或执行阶段立即失败,抛出RuntimeError

出错的代码(原始版本)

这是在修复之前,导致错误的测试代码结构和db_sessionfixture:

# test/services/test_retrieval_service.py@pytest.fixtureasyncdefdb_session():""" Creates a new database session for testing. """# 错误发生点:直接调用 get_async_engine(),它返回的是一个被缓存的 Engine 实例# 这个 Engine 绑定到了创建它时的 Event Loop(即第一个测试的 Loop)engine=get_async_engine()async_session=async_sessionmaker(engine,expire_on_commit=False)asyncwithasync_session()assession:asyncwithengine.begin()asconn:awaitconn.run_sync(Base.metadata.create_all)yieldsessionawaitsession.rollback()# 测试函数 1:使用新创建的 Loop A,成功获取 Engine(绑定到 Loop A)@pytest.mark.asyncioasyncdeftest_search_knowledge_base_flow(db_session):# ... PASS ...# 测试函数 2:使用新创建的 Loop B# 这里的 db_session fixture 再次运行,但 get_async_engine() 返回的是# 绑定到已关闭的 Loop A 的旧 Engine。导致报错。@pytest.mark.asyncioasyncdeftest_search_knowledge_base_no_results(db_session):# ... FAIL with RuntimeError: Event loop is closed ...

2. 根本原因分析

2.1 冲突来源

该问题源于pytest-asyncio管理 Event Loop 的机制与我们应用程序创建 SQLAlchemy Engine 的方式之间存在冲突。

  1. Pytest-Asyncio 的行为:默认情况下(严格模式),pytest-asyncio会为每个测试函数创建一个新的asyncio Event Loop,以确保隔离性。
  2. 应用程序的行为:我们的src/configs/db.py使用了functools.lru_cache来缓存AsyncEngine实例:
    # src/configs/db.pyfromfunctoolsimportlru_cache@lru_cache()# <--- Engine 实例被缓存了defget_async_engine():""" Returns a cached async engine instance. The engine is created on the first call and reused on subsequent calls within the same event loop. """logger.info("Creating new async engine instance.")returncreate_async_engine(DATABASE_URL,pool_pre_ping=True,echo=False,)

2.2 事件序列

  1. 测试 1 开始
    • Pytest 创建Loop A
    • get_async_engine()被调用。它创建了Engine 1并将其绑定到Loop A
    • 测试 1 结束。Pytest 关闭Loop A
  2. 测试 2 开始
    • Pytest 创建Loop B
    • get_async_engine()再次被调用。
    • 由于有缓存(@lru_cache),它返回了Engine 1(这个 Engine 仍然绑定在已关闭的Loop A上)。
    • 当 SQLAlchemy 尝试使用Engine 1Loop B中连接数据库或执行查询时,失败了,因为 Engine 的内部组件(如asyncpg连接池)试图使用已关闭的 Loop A。

3. 解决方案

3.1 修复方法

我们需要确保为每个测试上下文创建一个新的 AsyncEngine,并绑定到当前由pytest-asyncio提供的 Event Loop。

我们在测试文件 (test/services/test_retrieval_service.py) 的db_sessionfixture 中修改了代码,在请求 Engine 之前显式清除缓存。

@pytest.fixtureasyncdefdb_session():""" Creates a new database session for testing. """# 修复:强制为当前 Event Loop 创建一个新的 Engineget_async_engine.cache_clear()engine=get_async_engine()# 现在返回的是绑定到当前 Loop 的新 Engine# ... fixture 的其余部分 ...

3.2 为什么有效

通过调用get_async_engine.cache_clear(),我们使缓存的AsyncEngine实例失效。随后的get_async_engine()调用会重新执行函数体,创建一个正确绑定到当前运行 Event Loop 的新AsyncEngine实例。

4. 替代方案(供参考)

  1. Scope 匹配:将event_loopfixture 的 scope 更改为session(所有测试共用一个 Loop)。这虽然降低了隔离性,但避免了多 Loop 问题。
  2. 依赖覆盖:如果使用依赖注入框架,可以覆盖get_async_engine依赖。
  3. 全局 Conftest:在conftest.py的 autouse fixture 中实现缓存清除,从而全局应用于所有测试。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/30 22:02:02

运维新人必读:十大常见网络故障排查指南

一、网络故障排查基本原则在进入具体问题前&#xff0c;记住这三个核心原则&#xff1a;1. 从底层到高层&#xff1a;先物理层&#xff0c;再数据链路层&#xff0c;依次向上排查 2. 从简单到复杂&#xff1a;先检查最可能、最简单的因素 3. 变更回溯&#xff1a;最近有什么变动…

作者头像 李华
网站建设 2026/5/2 19:05:19

Cortex-M3中HardFault_Handler深度剖析:系统异常全面讲解

破解Cortex-M3的“死机之谜”&#xff1a;从HardFault到精准诊断你有没有遇到过这样的场景&#xff1f;设备在运行中突然“卡死”&#xff0c;LED停止闪烁&#xff0c;串口不再输出&#xff0c;调试器一连上却发现程序停在了一个叫HardFault_Handler的函数里——而你完全不知道…

作者头像 李华
网站建设 2026/5/1 18:46:39

uds31服务在Bootloader阶段的典型应用

uds31服务在Bootloader阶段的实战应用&#xff1a;从协议解析到工程落地当你在刷写ECU时&#xff0c;谁在幕后“点火”&#xff1f;你有没有想过&#xff0c;在整车厂产线或售后维修站执行一次固件刷新时&#xff0c;为什么不是一上电就直接开始烧录&#xff1f;为什么诊断工具…

作者头像 李华
网站建设 2026/5/1 18:47:30

MOSFET高边驱动自举二极管选型全面讲解

深入理解MOSFET高边驱动&#xff1a;自举二极管为何如此关键&#xff1f;在设计一个高效、可靠的DC-DC变换器或电机驱动电路时&#xff0c;你是否曾遇到过这样的问题&#xff1a;高边MOSFET总是无法完全导通&#xff1f;系统发热严重&#xff1f;甚至在高温下直接“丢脉冲”导致…

作者头像 李华
网站建设 2026/5/1 7:55:15

Miniconda-Python3.10镜像在语音合成大模型中的实践

Miniconda-Python3.10镜像在语音合成大模型中的实践 在当前AI研发节奏日益加快的背景下&#xff0c;语音合成技术正从实验室走向大规模落地。无论是智能音箱里的自然对话&#xff0c;还是有声书平台上的拟人朗读&#xff0c;背后都离不开高质量TTS模型的支持。但一个常被忽视的…

作者头像 李华
网站建设 2026/5/4 22:30:30

STM32中hal_uart_transmit的入门操作指南

从零开始掌握 STM32 串口发送&#xff1a; HAL_UART_Transmit 实战全解析 在嵌入式开发的日常中&#xff0c;你有没有遇到过这样的场景&#xff1f;代码烧录成功、板子通电正常&#xff0c;但调试助手却迟迟没有输出“Hello World”——那一刻&#xff0c;是不是怀疑人生了&a…

作者头像 李华