当前位置: 首页 > article >正文

自动化测试(十) 微服务测试策略-单元到集成到契约到端到端分层实战

微服务测试策略单元→集成→契约→端到端分层实战前面咱们分别聊了单元测试、接口测试、契约测试。今天把它们串起来聊聊微服务架构下怎么设计完整的测试策略——每一层测什么、怎么测、用什么工具。一、微服务测试的金字塔变体单体应用的测试金字塔在微服务架构下要调整单体应用金字塔 微服务金字塔 /\ /\ / \ E2E少 / \ 端到端极少 /----\ /----\ ← 只验证核心链路 / \ 集成中 / \ 契约测试中 /--------\ /--------\ ← 服务间接口约定 / \ 单元多 / \ 集成单元多 /------------\ /------------\ ← 服务内部分层关键变化单元测试依然最多但增加了服务内集成测试契约测试成为新的一层验证服务间接口端到端测试更少因为涉及多个服务成本高、定位难二、分层实战从里到外假设咱们要测试一个订单服务它有这些组件┌─────────────────────────────────────────┐ │ OrderController │ ← REST API入口 │ POST /orders GET /orders/{id} │ ├─────────────────────────────────────────┤ │ OrderService │ ← 业务逻辑 │ createOrder() getOrder() │ ├─────────────────────────────────────────┤ │ OrderRepository │ ← 数据访问 │ Spring Data JPA │ ├─────────────────────────────────────────┤ │ 外部依赖 │ │ UserClient StockClient PayClient │ ← Feign/HTTP调用 └─────────────────────────────────────────┘第一层单元测试Unit Test测什么OrderService的业务逻辑不依赖Spring容器、不连数据库、不调外部服务。ExtendWith(MockitoExtension.class)classOrderServiceTest{MockOrderRepositoryorderRepository;MockUserClientuserClient;MockStockClientstockClient;MockPayClientpayClient;MockApplicationEventPublishereventPublisher;InjectMocksOrderServiceorderService;TestDisplayName(正常创建订单扣库存、算价格、发事件)voidshouldCreateOrderSuccessfully(){// GivenCreateOrderRequestrequestnewCreateOrderRequest(ITEM-001,2);when(userClient.getCurrentUser()).thenReturn(newUser(1L,Alice));when(stockClient.deduct(ITEM-001,2)).thenReturn(true);when(orderRepository.save(any())).thenAnswer(inv-{Orderorderinv.getArgument(0);order.setId(100L);returnorder;});// WhenOrderResultresultorderService.createOrder(request);// ThenassertThat(result.isSuccess()).isTrue();assertThat(result.getOrderId()).isEqualTo(100L);// 验证交互verify(stockClient).deduct(ITEM-001,2);verify(orderRepository).save(argThat(order-order.getTotalAmount().equals(newBigDecimal(199.98))));verify(eventPublisher).publishEvent(any(OrderCreatedEvent.class));}TestDisplayName(库存不足时订单创建失败)voidshouldFailWhenStockInsufficient(){when(userClient.getCurrentUser()).thenReturn(newUser(1L,Alice));when(stockClient.deduct(ITEM-001,100)).thenReturn(false);assertThatThrownBy(()-orderService.createOrder(newCreateOrderRequest(ITEM-001,100))).isInstanceOf(InsufficientStockException.class);verify(orderRepository,never()).save(any());}}单元测试的关键快毫秒级独立不依赖外部覆盖分支if/else、异常路径第二层切片测试Slice TestSpring Boot提供了WebMvcTest、DataJpaTest等注解只加载特定层的Bean。WebMvcTest只测Controller层WebMvcTest(OrderController.class)AutoConfigureMockMvcclassOrderControllerTest{AutowiredMockMvcmockMvc;MockBeanOrderServiceorderService;MockBeanJwtTokenProvidertokenProvider;AutowiredObjectMapperobjectMapper;TestDisplayName(创建订单接口返回201和订单ID)voidshouldReturn201WhenCreateOrder()throwsException{when(orderService.createOrder(any())).thenReturn(newOrderResult(true,100L,CREATED));mockMvc.perform(post(/api/orders).contentType(MediaType.APPLICATION_JSON).header(Authorization,Bearer valid-token).content(objectMapper.writeValueAsString(Map.of(sku,ITEM-001,quantity,2)))).andExpect(status().isCreated()).andExpect(jsonPath($.success).value(true)).andExpect(jsonPath($.orderId).value(100));}TestDisplayName(缺少必填参数返回400)voidshouldReturn400WhenMissingRequiredField()throwsException{mockMvc.perform(post(/api/orders).contentType(MediaType.APPLICATION_JSON).header(Authorization,Bearer valid-token).content({}))// 空body.andExpect(status().isBadRequest()).andExpect(jsonPath($.errors).isArray());}}WebMvcTest只加载Controller、ControllerAdvice、JsonComponentConverter、Filter、WebMvcConfigurer不加载Service、Repository、Component启动速度比SpringBootTest快得多。DataJpaTest只测Repository层DataJpaTestAutoConfigureTestDatabase(replaceAutoConfigureTestDatabase.Replace.NONE)TestcontainersclassOrderRepositoryTest{ContainerstaticMySQLContainer?mysqlnewMySQLContainer(mysql:8.0);DynamicPropertySourcestaticvoidconfigureProperties(DynamicPropertyRegistryregistry){registry.add(spring.datasource.url,mysql::getJdbcUrl);}AutowiredOrderRepositoryorderRepository;TestDisplayName(根据用户ID查询订单)voidshouldFindOrdersByUserId(){// GivenorderRepository.save(newOrder(1L,ITEM-001,newBigDecimal(99.99)));orderRepository.save(newOrder(1L,ITEM-002,newBigDecimal(199.99)));orderRepository.save(newOrder(2L,ITEM-003,newBigDecimal(50.00)));// WhenListOrderuser1OrdersorderRepository.findByUserId(1L);// ThenassertThat(user1Orders).hasSize(2);assertThat(user1Orders).extracting(Order::getSku).containsExactlyInAnyOrder(ITEM-001,ITEM-002);}TestDisplayName(复杂查询查询某用户的待支付订单总金额)voidshouldCalculatePendingAmount(){// 测试Query注解写的复杂SQLBigDecimaltotalorderRepository.calculatePendingAmountByUserId(1L);assertThat(total).isEqualByComparingTo(newBigDecimal(299.98));}}DataJpaTest只加载Entity、Repository嵌入式数据库默认或配置的数据源不加载Controller、Service第三层服务内集成测试Integration Test测整个服务但Mock外部依赖。SpringBootTest(webEnvironmentSpringBootTest.WebEnvironment.RANDOM_PORT)TestcontainersclassOrderServiceIntegrationTest{LocalServerPortintport;ContainerstaticMySQLContainer?mysqlnewMySQLContainer(mysql:8.0);RegisterExtensionstaticWireMockExtensionuserServiceMockWireMockExtension.newInstance().options(wireMockConfig().dynamicPort()).build();RegisterExtensionstaticWireMockExtensionstockServiceMockWireMockExtension.newInstance().options(wireMockConfig().dynamicPort()).build();DynamicPropertySourcestaticvoidconfigureProperties(DynamicPropertyRegistryregistry){registry.add(spring.datasource.url,mysql::getJdbcUrl);registry.add(services.user.url,userServiceMock::baseUrl);registry.add(services.stock.url,stockServiceMock::baseUrl);}AutowiredTestRestTemplaterestTemplate;AutowiredOrderRepositoryorderRepository;BeforeEachvoidsetUp(){orderRepository.deleteAll();// 配置下游MockuserServiceMock.stubFor(get(/api/users/current).willReturn(okJson({id:1,name:Alice})));stockServiceMock.stubFor(post(/api/stock/deduct).willReturn(okJson({success:true,remaining:98})));}TestDisplayName(完整流程创建订单并持久化到数据库)voidshouldCreateOrderAndPersist(){// GivenCreateOrderRequestrequestnewCreateOrderRequest(ITEM-001,2);// WhenResponseEntityOrderResultresponserestTemplate.postForEntity(/api/orders,request,OrderResult.class);// Then: HTTP响应assertThat(response.getStatusCode()).isEqualTo(HttpStatus.CREATED);assertThat(response.getBody().isSuccess()).isTrue();// Then: 数据库验证ListOrderordersorderRepository.findAll();assertThat(orders).hasSize(1);assertThat(orders.get(0).getSku()).isEqualTo(ITEM-001);assertThat(orders.get(0).getStatus()).isEqualTo(OrderStatus.CREATED);// Then: 下游服务确实被调用了stockServiceMock.verify(postRequestedFor(urlEqualTo(/api/stock/deduct)));}}第四层契约测试Contract Test前面第5篇详细讲过这里总结在分层中的位置// 消费者端订单服务验证契约PactTestFor(providerNamestock-service)classStockServiceContractTest{Pact(consumerorder-service)RequestResponsePactstockDeductPact(PactDslWithProviderbuilder){returnbuilder.given(stock available).uponReceiving(deduct stock).method(POST).path(/api/stock/deduct).body(newPactDslJsonBody().stringType(sku,ITEM-001).integerType(quantity,2)).willRespondWith().status(200).body(newPactDslJsonBody().booleanType(success,true).integerType(remaining,98)).toPact();}}第五层端到端测试E2E Test只验证最核心的用户路径涉及多个真实服务。// 用Docker Compose启动所有服务然后跑E2E测试TestcontainersclassOrderE2ETest{ContainerstaticDockerComposeContainer?environmentnewDockerComposeContainer(newFile(docker-compose.test.yml)).withExposedService(order-service,8080).withExposedService(user-service,8080).withExposedService(stock-service,8080).withExposedService(mysql,3306).waitingFor(order-service,Wait.forHttp(/actuator/health).forStatusCode(200));TestDisplayName(用户下单完整流程)voidshouldCompleteOrderFlow(){StringorderServiceUrlhttp://environment.getServiceHost(order-service,8080):environment.getServicePort(order-service,8080);// 1. 用户注册// 2. 用户登录// 3. 浏览商品// 4. 创建订单// 5. 支付订单// 6. 验证订单状态given().baseUri(orderServiceUrl).body(createOrderRequest).when().post(/api/orders).then().statusCode(201).body(status,equalTo(PAID));}}三、各层的执行策略# CI流水线分层执行stages:-unit-tests# 每次提交都跑 1分钟-integration-tests# 每次提交都跑 5分钟-contract-tests# 每次提交都跑 2分钟-e2e-tests# 合并到main分支前跑 15分钟unit-tests:script:-mvn test-pl order-service# 只跑单元测试only:-merge_requests-mainintegration-tests:script:-mvn verify-Pintegration-test# 跑集成测试only:-merge_requests-maine2e-tests:script:-docker-compose-f docker-compose.test.yml up-d-mvn test-Pe2e-test-docker-compose-f docker-compose.test.yml downonly:-main# 只在main分支跑when:manual# 可手动触发四、小结今天咱们串起了微服务测试的完整分层层级范围工具速度频率单元测试Service/UtilJUnit Mockito毫秒每次提交切片测试Controller/RepositoryWebMvcTest / DataJpaTest秒每次提交集成测试整个服务SpringBootTest Testcontainers10秒每次提交契约测试服务间接口Pact / Spring Cloud Contract秒每次提交E2E测试多服务完整流程Docker Compose RestAssured分钟合并前一句话总结微服务测试不是测得越多越好而是每层测该测的。单元测逻辑、集成测协作、契约测约定、E2E验流程。

相关文章:

自动化测试(十) 微服务测试策略-单元到集成到契约到端到端分层实战

微服务测试策略:单元→集成→契约→端到端分层实战前面咱们分别聊了单元测试、接口测试、契约测试。今天把它们串起来,聊聊微服务架构下怎么设计完整的测试策略——每一层测什么、怎么测、用什么工具。一、微服务测试的"金字塔"变体 单体应用的…...

蓝牙窃密攻防实战:从协议漏洞到固件后门,国家安全部警示的近场威胁全解析

2026年5月11日,国家安全部官方发布重磅警示,明确指出蓝牙设备已成为不法分子实施近距离窃密、监听、跟踪的"隐形獠牙"。从日常使用的无线耳机、智能手表,到办公场景的蓝牙键鼠、会议音箱,再到工业控制中的蓝牙传感器&am…...

芯片设计公司ISO 9001认证:从质量管理体系到流片成功的工程实践

1. 从一则旧闻聊起:ISO 9001认证对一家芯片设计公司意味着什么?前几天在整理资料时,偶然翻到一篇2011年的行业旧闻,说的是当时一家名为SiliconBlue Technologies的公司,获得了ISO 9001:2008质量管理体系认证。新闻稿写…...

音频算法调试利器:用Android App实时绘制EQ/DRC曲线,告别Matlab依赖

移动端音频算法调试革命:Android实时EQ/DRC可视化工具开发实战 在音频算法开发领域,调试环节长期被桌面级工具垄断,工程师们不得不忍受开发板与工作站之间的频繁切换。这种工作模式不仅效率低下,更无法满足现代音频产品快速迭代的…...

AntiDupl.NET:告别数字杂乱,让图片管理回归优雅

AntiDupl.NET:告别数字杂乱,让图片管理回归优雅 【免费下载链接】AntiDupl A program to search similar and defect pictures on the disk 项目地址: https://gitcode.com/gh_mirrors/an/AntiDupl 你是否曾经在整理照片时,发现手机里…...

基于本地大模型与Playwright的隐私优先求职自动化助手RedClaw实践

1. 项目概述:一个真正为你掌控的本地化求职AI助手在求职季,我们常常面临一个两难困境:一方面,海投简历耗时耗力,重复填写那些大同小异的在线申请表让人筋疲力尽;另一方面,市面上一些所谓的“自动…...

苹果为何拒绝TD-SCDMA特供版iPhone?复盘技术标准与市场时机的战略博弈

1. 项目概述:一场关于苹果与中国移动的世纪猜想2012年的科技圈,空气中弥漫着一股躁动与期待。几乎所有的行业分析师和手机发烧友都在讨论同一个话题:苹果公司是否会为了全球最大的移动运营商——中国移动,专门推出一款支持TD-SCDM…...

机器视觉在人工智能领域的应用

机器视觉在人工智能领域的应用 目录机器视觉在人工智能领域的应用一、图像处理与机器视觉的概念阐述1. 图像处理(Image Processing)2. 机器视觉(Machine Vision / Computer Vision)二、图像处理与机器视觉的区别与共同点区别共同点…...

如何通过Python快速接入Taotoken并调用多模型API完成文本生成任务

🚀 告别海外账号与网络限制!稳定直连全球优质大模型,限时半价接入中。 👉 点击领取海量免费额度 如何通过Python快速接入Taotoken并调用多模型API完成文本生成任务 1. 准备工作:获取API Key与模型ID 在开始编写代码之…...

嵌入式开发中的编程规范实践与行业标准解析

1. 编程规范的本质与价值在嵌入式汽车电子领域干了十五年,我见过太多因为代码不规范导致的惨痛教训。有一次,某车企的ECU控制模块在零下30度环境突然死机,排查三周后发现是未初始化的指针在低温环境下产生了非预期行为——这种问题本可以通过…...

实战复盘:我是如何通过一个SSRF漏洞,利用Gopher协议拿下内网Redis的

从SSRF到内网Redis入侵:一次真实渗透测试的深度剖析 那天下午,我正在对某企业Web应用进行常规安全评估。一个看似普通的文件下载接口引起了我的注意——它接受URL参数并返回对应资源内容。直觉告诉我,这里可能存在SSRF漏洞。接下来的72小时&a…...

一句话就能“劫持”你的AI?DZS 分层式自适应提示词注入攻击的防御机制框架 (HAA)来了!

本文所展示的提示词技术已在Research square 发表论文预印本。DOI:https://doi.org/10.21203/rs.3.rs-9653510/v1 作者“抖知书(douzhishu),涉及到相关测试数据是本人自行测试的,并未通过多专家评审,所以仅…...

新手避坑指南:用Simulink搭建48V开关电源仿真,从整流到反激电路完整流程

新手避坑指南:用Simulink搭建48V开关电源仿真全流程实战 电力电子领域的仿真实验常常让初学者望而生畏——参数设置不当可能导致虚拟元器件"烧毁",波形失真却找不到原因。本文将手把手带你用Simulink搭建从交流整流到DC-DC变换的完整48V电源系…...

PX4倾转垂起固定翼混控配置与硬件适配实战

1. PX4倾转垂起固定翼的核心概念解析 第一次接触倾转垂起固定翼的朋友可能会被这个名词吓到,其实它的原理并不复杂。简单来说,这是一种既能像多旋翼一样垂直起降,又能像固定翼飞机一样高效巡航的混合飞行器。我经手过的项目中,这种…...

告别公网IP焦虑:用SakuraFrp免费隧道,5分钟搞定Linux服务器的SSH远程访问

5分钟实现无公网IP的Linux服务器远程访问:SakuraFrp实战指南 当你需要在外紧急处理家中或办公室的Linux服务器时,却发现没有公网IP无法远程连接,这种焦虑我深有体会。去年深夜的一次线上故障让我深刻认识到内网穿透工具的重要性——当时我正…...

SBQE:量子机器学习数据编码的创新方法

1. SBQE:量子机器学习数据编码的新范式量子计算领域最近迎来了一项突破性进展——SBQE(Shot-Based Quantum Encoding)数据编码方法。作为一名长期跟踪量子机器学习发展的研究者,我亲历了这项技术从理论提出到实验验证的全过程。SB…...

R3nzSkin英雄联盟皮肤修改器完整教程:免费体验全皮肤的终极指南

R3nzSkin英雄联盟皮肤修改器完整教程:免费体验全皮肤的终极指南 【免费下载链接】R3nzSkin Skin changer for League of Legends (LOL) 项目地址: https://gitcode.com/gh_mirrors/r3n/R3nzSkin R3nzSkin是一款专为《英雄联盟》玩家设计的开源皮肤修改工具&a…...

onlybooks/llm项目解析:大语言模型本地部署与微调实战指南

1. 项目概述与核心价值最近在折腾大语言模型本地部署和微调的朋友,估计没少在各种开源社区和模型仓库里翻找。我自己也是,从早期的GPT-2到现在的各种百亿、千亿参数模型,一路踩坑过来,深感一个清晰、易用、维护良好的项目对效率提…...

从零上手CircuitJS1:开源电路仿真工具的核心功能与实战演练

1. 初识CircuitJS1:浏览器里的电子实验室 第一次打开CircuitJS1时,我仿佛回到了大学电子实验室——只不过这次所有仪器都装进了浏览器窗口。这个完全开源的工具用JavaScript重构了经典的Falstad电路模拟器,不需要安装任何插件就能在Chrome或…...

缠论可视化插件:3个步骤让你的技术分析不再迷茫

缠论可视化插件:3个步骤让你的技术分析不再迷茫 【免费下载链接】Indicator 通达信缠论可视化分析插件 项目地址: https://gitcode.com/gh_mirrors/ind/Indicator 你是否曾经面对复杂的K线图感到无从下手?缠论作为技术分析的重要理论,…...

如何用wxlivespy实现微信视频号直播数据实时抓取与分析

如何用wxlivespy实现微信视频号直播数据实时抓取与分析 【免费下载链接】wxlivespy 微信视频号直播间弹幕信息抓取工具 项目地址: https://gitcode.com/gh_mirrors/wx/wxlivespy wxlivespy是一款专业级的微信视频号直播间弹幕信息抓取工具,能够实时捕获弹幕、…...

Hydrin 1 ([Arg8, Gly10, Lys11, Arg12]-Vasotocin)

一、基础信息多肽名称:Hydrin 1,加压催产素变体 [Arg8, Gly10, Lys11, Arg12]-Vasotocin 三字母序列:Cys-Tyr-Ile-Gln-Asn-Cys-Pro-Arg-Gly-Gly-Lys-Arg 单字母序列:CYIQNCPRG GKR 氨基酸数目:12 aa 结构特征&#xff…...

Hummingbot自动化交易框架:从原理到实战的量化交易指南

1. 项目概述:一个为专业交易者打造的自动化交易框架如果你在加密货币交易领域摸爬滚打过一段时间,一定会对“手动盯盘”的疲惫和“情绪化操作”的代价深有体会。市场24/7运转,机会转瞬即逝,而人的精力终究有限。这正是我最初接触并…...

Pearcleaner:macOS终极免费应用清理工具,彻底告别数字残留

Pearcleaner:macOS终极免费应用清理工具,彻底告别数字残留 【免费下载链接】Pearcleaner A free, source-available and fair-code licensed mac app cleaner 项目地址: https://gitcode.com/gh_mirrors/pe/Pearcleaner 你是否曾经在macOS上删除应…...

本地大模型无缝集成IDE:TRAE-Ollama-Bridge透明代理方案详解

1. 项目概述与核心痛点 如果你和我一样,是个喜欢在本地折腾大模型的开发者,那你肯定对 Ollama 不陌生。它能让我们在个人电脑上轻松运行 Llama、Qwen、DeepSeek 这些开源模型,速度快,隐私好,还不用花 API 调用费。但问…...

开源游戏串流革命:Sunshine如何重新定义家庭游戏共享体验

开源游戏串流革命:Sunshine如何重新定义家庭游戏共享体验 【免费下载链接】Sunshine Self-hosted game stream host for Moonlight. 项目地址: https://gitcode.com/GitHub_Trending/su/Sunshine 在游戏设备日益多样化的今天,你是否曾想过将高性能…...

ProxyClaw住宅代理实战:破解反爬虫,赋能AI智能体与数据工程

1. 项目概述:ProxyClaw,一个为AI与数据工程而生的住宅代理网络 如果你正在构建一个需要从互联网上大规模、稳定抓取数据的AI智能体、自动化机器人或者数据管道,那么“被目标网站封禁”这件事,大概率是你最头疼的日常。无论是电商平…...

别再被Linux的free命令骗了!手把手教你读懂‘可用内存’和‘实际空闲内存’的区别

别再被Linux的free命令骗了!手把手教你读懂‘可用内存’和‘实际空闲内存’的区别 刚接触Linux服务器管理时,看到free -m输出里那个触目惊心的"free"数值,我的第一反应是:"天哪,内存快用完了&#xff0…...

别再手动写Prompt了!Lovable原生AI编排引擎深度解析(附12个已验证行业工作流)

更多请点击: https://intelliparadigm.com 第一章:Lovable无代码AI应用构建指南 Lovable 是一款面向业务人员与开发者的低门槛 AI 应用构建平台,它通过可视化编排、预置模型组件和自然语言驱动逻辑,实现无需编写代码即可部署可运…...

搞懂VMware三种网络模式:从NAT断网到桥接、仅主机的实战选择指南

VMware虚拟网络模式深度解析:从原理到场景化实战 当你在深夜赶项目时,虚拟机突然无法联网——这种经历对于开发者而言无异于噩梦。我曾亲眼见过团队新人在演示前夜因NAT模式配置问题崩溃,也见证过安全工程师因选错网络模式导致测试环境暴露。…...