三十九、【扩展工具篇】Allpairspy 组合用例生成器:智能设计高效测试集
三十九、【扩展工具篇】Allpairspy 组合用例生成器:智能设计高效测试集
-
- 前言
-
- 准备工作
- 第一部分:后端实现 - `allpairspy` API
-
- 1. 创建 `allpairspy` 服务
- 2. 创建 `allpairspy` API 视图
- 3. 注册 API 路由
- 第二部分:前端实现 - `Allpairspy` 工具界面
-
- 1. 创建 API 服务 (`src/api/tools.ts`)
- 2. 添加路由和侧边栏菜单入口
- 3. 创建 `Allpairspy` 工具页面 (`src/views/tools/AllpairsGeneratorView.vue`)
- 第三部分:全面测试与验证
- 总结
前言
假设我们有一个功能,它有 3 个参数,每个参数有 3 个取值。如果进行全组合(笛卡尔积)测试,需要 3 * 3 * 3 = 27
个测试用例。而大多数由参数组合引发的缺陷,都是由两个参数的特定组合导致的。
All-Pairs 测试的核心思想就是:用最少的测试用
例,覆盖任意两个参数的所有取值组合。
例如,对于以下参数:
- 浏览器: Chrome, Firefox
- 操作系统: Windows, MacOS
- 用户状态: LoggedIn, Guest
全组合是 8 个用例,而 All-Pairs 算法只需要 4 个用例就能覆盖所有“两两组合”:
准备工作
- 前端项目就绪:
test-platform/frontend
项目可以正常运行 (npm run dev
)。 - 后端 API 运行中: Django 后端服务运行。
- Element Plus 集成完毕。
- 安装
allpairspy
库 (后端):
在你的 Django 项目的虚拟环境中运行:pip install allpairspy
第一部分:后端实现 - allpairspy
API
1. 创建 allpairspy
服务
a. 创建 api/services/allpairs_generator.py
文件:
b. 编写 allpairs_generator.py
:
# test-platform/api/services/allpairs_generator.py
from allpairspy import AllPairs
from typing import List, Anydef generate_allpairs_cases(parameters: List[List[Any]]) -> List[List[Any]]:"""使用 allpairspy 库生成组合测试用例。:param parameters: 二维列表,每个子列表是一个参数的所有取值。例如: [["Chrome", "Firefox"], ["Windows", "MacOS"]]:return: 一个包含所有组合用例的二维列表。"""if not parameters or not all(isinstance(p, list) and p for p in parameters):raise ValueError("输入参数必须是一个非空的二维列表,且每个子列表不能为空。")# AllPairs 是一个生成器,我们需要将其转换为列表return [list(case) for case in AllPairs(parameters)]
2. 创建 allpairspy
API 视图
打开 test-platform/api/views.py
,添加一个新的 APIView
。
# test-platform/api/views.py
from rest_framework.views import APIView
from rest_framework.response import Response
from rest_framework import permissions, status as http_status
from .services.allpairs_generator import generate_allpairs_cases # 导入服务函数
import logging # 导入 logginglogger = logging.getLogger(__name__)# ... (其他 ViewSet) ...class AllpairsGenerateView(APIView):"""Allpairspy 组合用例生成 APIPOST /api/tools/generate-allpairs/"""permission_classes = [permissions.IsAuthenticated] # 只有登录用户才能使用def post(self, request, *args, **kwargs):parameters_data = request.data.get('parameters', [])if not isinstance(parameters_data, list) or not parameters_data:return Response({"detail": "parameters 字段必须是一个非空列表。"}, status=http_status.HTTP_400_BAD_REQUEST)headers = []values_list = []for param in parameters_data:if not isinstance(param,