0

0

Jest 中为每个 it() 测试用例配置独立的模块 mock 实现

心靈之曲

心靈之曲

发布时间:2026-02-02 14:08:02

|

828人浏览过

|

来源于php中文网

原创

Jest 中为每个 it() 测试用例配置独立的模块 mock 实现

在 jest 中,`jest.mock()` 无法在 `it()` 内部动态生效;正确做法是顶层调用 `jest.mock()` 启用自动模拟,再结合 `jest.spyon()` + `mockimplementation()` 在每个测试中覆盖具体行为,并通过 `aftereach` 清理状态,确保测试隔离。

要为每个 it() 测试用例提供完全独立、互不干扰的模块 mock 实现(例如不同行为的中间件),关键在于理解 Jest 模块模拟的生命周期和隔离机制。直接在 it() 内调用 jest.mock() 是无效的——Jest 仅在模块加载时(即文件顶部)处理 jest.mock() 调用,运行时调用会被忽略。

✅ 正确且推荐的三步策略如下:

1. 顶层 jest.mock() 启用自动模拟

在文件顶部(describe 外或 describe 内顶部)调用 jest.mock(),启用对目标模块的自动模拟(auto-mock)。这会生成一个带默认 jest.fn() 行为的模拟模块,但不定义具体实现

// test.js —— 文件顶部
jest.mock('../../../middleware/awsTransferMiddleware');

const awsTransferMiddleware = require('../../../middleware/awsTransferMiddleware');
⚠️ 注意:必须 require()(而非 import)该模块,否则 jest.spyOn() 将无法访问其属性(ESM 静态导入在 Jest 中不支持运行时重写)。

2. 使用 jest.spyOn() 动态覆盖方法实现

在每个 it() 中,使用 jest.spyOn(对象, 方法名) 获取对模拟函数的引用,并调用 .mockImplementation() 设置当前测试专属逻辑

it("Should return 200 if at least one image exists", async () => {
  // ✅ 覆盖 transferS3Files:模拟成功上传
  jest.spyOn(awsTransferMiddleware, 'transferS3Files')
    .mockImplementation(async (req, res, next) => {
      req.s3TransferResult = { success: true, count: 2 };
      next();
    });

  // ✅ 覆盖 filterPassedImage:模拟筛选出 1 张有效图
  jest.spyOn(awsTransferMiddleware, 'filterPassedImage')
    .mockImplementation(async (req, res, next) => {
      req.filteredImages = ['img1.jpg'];
      next();
    });

  const response = await request(app).post('/upload');
  expect(response.status).toBe(200);
});
it("Should return 400 when S3 upload fails", async () => {
  // ❌ 完全不同的行为:模拟 transferS3Files 抛错
  jest.spyOn(awsTransferMiddleware, 'transferS3Files')
    .mockImplementation(async (req, res, next) => {
      const err = new Error('S3 timeout');
      err.status = 503;
      next(err);
    });

  // ✅ filterPassedImage 不执行(因 transferS3Files 已调用 next(err))
  jest.spyOn(awsTransferMiddleware, 'filterPassedImage')
    .mockImplementation(() => {
      throw new Error('This should not be called');
    });

  const response = await request(app).post('/upload');
  expect(response.status).toBe(400); // 假设错误被全局 handler 转换为 400
});

3. afterEach 全面清理,保障测试纯净性

在每个测试结束后,清除所有 mock 状态,防止“泄漏”到下一个用例:

Calliper 文档对比神器
Calliper 文档对比神器

文档内容对比神器

下载
afterEach(() => {
  jest.restoreAllMocks(); // 恢复所有被 spy 的原始方法(如需)
  jest.clearAllMocks();   // 清空所有 mock 函数的调用记录与返回值
});

✅ jest.clearAllMocks() 是核心:它重置 mockImplementation、mockReturnValue、调用计数等,使下个 it() 可安全重新设置。

? 为什么不用 jest.resetModules()?

jest.resetModules() 会重新 require 模块,但对已 require 的模块实例(如 app 或 request 中依赖的中间件)无效,且可能破坏 supertest 的应用实例缓存。而 spyOn + clearAllMocks 更精准、轻量、可靠。

✅ 最终完整结构示例

// test.js
const request = require('supertest');
const app = require('../../../path/to/your/app');

// Step 1: Top-level mock
jest.mock('../../../middleware/awsTransferMiddleware');
const awsTransferMiddleware = require('../../../middleware/awsTransferMiddleware');

// Step 2: Cleanup after each test
afterEach(() => {
  jest.restoreAllMocks();
  jest.clearAllMocks();
});

describe('POST /upload', () => {
  it('returns 200 on successful transfer and filtering', async () => {
    jest.spyOn(awsTransferMiddleware, 'transferS3Files')
      .mockImplementation(async (req, res, next) => {
        req.s3TransferResult = { success: true };
        next();
      });
    jest.spyOn(awsTransferMiddleware, 'filterPassedImage')
      .mockImplementation(async (req, res, next) => {
        req.filteredImages = ['a.jpg'];
        next();
      });

    const res = await request(app).post('/upload');
    expect(res.status).toBe(200);
  });

  it('returns 500 when S3 transfer throws', async () => {
    jest.spyOn(awsTransferMiddleware, 'transferS3Files')
      .mockImplementation(async () => {
        throw new Error('Network error');
      });

    const res = await request(app).post('/upload');
    expect(res.status).toBe(500);
  });
});

? 总结:Jest 的模块 mock 隔离不靠“多次 jest.mock()”,而靠 “一次模拟 + 多次 spyOn().mockImplementation() + 每次 clearAllMocks()”。这是官方推荐、稳定高效、符合测试隔离原则的最佳实践。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
什么是中间件
什么是中间件

中间件是一种软件组件,充当不兼容组件之间的桥梁,提供额外服务,例如集成异构系统、提供常用服务、提高应用程序性能,以及简化应用程序开发。想了解更多中间件的相关内容,可以阅读本专题下面的文章。

178

2024.05.11

Golang 中间件开发与微服务架构
Golang 中间件开发与微服务架构

本专题系统讲解 Golang 在微服务架构中的中间件开发,包括日志处理、限流与熔断、认证与授权、服务监控、API 网关设计等常见中间件功能的实现。通过实战项目,帮助开发者理解如何使用 Go 编写高效、可扩展的中间件组件,并在微服务环境中进行灵活部署与管理。

217

2025.12.18

require的用法
require的用法

require的用法有引入模块、导入类或方法、执行特定任务。想了解更多require的相关内容,可以阅读本专题下面的文章。

467

2023.11.27

AO3官网入口与中文阅读设置 AO3网页版使用与访问
AO3官网入口与中文阅读设置 AO3网页版使用与访问

本专题围绕 Archive of Our Own(AO3)官网入口展开,系统整理 AO3 最新可用官网地址、网页版访问方式、正确打开链接的方法,并详细讲解 AO3 中文界面设置、阅读语言切换及基础使用流程,帮助用户稳定访问 AO3 官网,高效完成中文阅读与作品浏览。

19

2026.02.02

主流快递单号查询入口 实时物流进度一站式追踪专题
主流快递单号查询入口 实时物流进度一站式追踪专题

本专题聚合极兔快递、京东快递、中通快递、圆通快递、韵达快递等主流物流平台的单号查询与运单追踪内容,重点解决单号查询、手机号查物流、官网入口直达、包裹进度实时追踪等高频问题,帮助用户快速获取最新物流状态,提升查件效率与使用体验。

6

2026.02.02

Golang WebAssembly(WASM)开发入门
Golang WebAssembly(WASM)开发入门

本专题系统讲解 Golang 在 WebAssembly(WASM)开发中的实践方法,涵盖 WASM 基础原理、Go 编译到 WASM 的流程、与 JavaScript 的交互方式、性能与体积优化,以及典型应用场景(如前端计算、跨平台模块)。帮助开发者掌握 Go 在新一代 Web 技术栈中的应用能力。

1

2026.02.02

PHP Swoole 高性能服务开发
PHP Swoole 高性能服务开发

本专题聚焦 PHP Swoole 扩展在高性能服务端开发中的应用,系统讲解协程模型、异步IO、TCP/HTTP/WebSocket服务器、进程与任务管理、常驻内存架构设计。通过实战案例,帮助开发者掌握 使用 PHP 构建高并发、低延迟服务端应用的工程化能力。

2

2026.02.02

Java JNI 与本地代码交互实战
Java JNI 与本地代码交互实战

本专题系统讲解 Java 通过 JNI 调用 C/C++ 本地代码的核心机制,涵盖 JNI 基本原理、数据类型映射、内存管理、异常处理、性能优化策略以及典型应用场景(如高性能计算、底层库封装)。通过实战示例,帮助开发者掌握 Java 与本地代码混合开发的完整流程。

1

2026.02.02

go语言 注释编码
go语言 注释编码

本专题整合了go语言注释、注释规范等等内容,阅读专题下面的文章了解更多详细内容。

61

2026.01.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.7万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号