将分散的Pytest测试脚本统一接入测试平台:FastAPI改造方案详解

在上一篇文章《Pytest 测试用例自动生成:接口自动化进阶实践》中,我们已经解决了“如何高效编写和维护接口自动化用例”的问题。

然而,随着业务的发展和团队规模的扩大,很多公司会选择开发自己的测试平台,以实现更高效、更统一的管理。

企业接口自动化通常会经历如下过程:

  1. 初期:测试人员本地用 Pytest 写脚本(即脚本项目)
  2. 中期:接入 Jenkins,定时跑一跑
  3. 后期:公司开始建设统一测试平台

一. 为什么需要将脚本项目接入测试平台

1. 当前面临的挑战

  • 用例复用难:已有的大量 Pytest 测试用例无法直接在平台上运行,重新编写成本高。
  • 管理复杂:多个脚本项目分散管理,执行效率低、维护困难。
  • 报告分散:测试报告散落在各个项目中,难以统一查看和分析。

因此,需要考虑在【不重写、不侵入、不破坏】现有 Pytest 脚本项目的前提下,让它具备“测试平台可接入能力”

2. 两种可选方案

我们考虑以下两种方案将脚本项目接入测试平台:

  • 方案一:将脚本项目直接集成到测试平台目录中。
  • 方案二:使用 FastAPI 改造脚本项目,提供接口供平台调用。

为什么选择方案二?

  • 脚本项目经常需要修改和更新,若集成到平台中,每次修改都需要重新发布平台代码,繁琐且易出错。
  • FastAPI 轻量、高性能,适合快速构建RESTful接口,实现脚本与平台的解耦。

二. 使用 FastAPI 框架改造脚本项目

1. 核心思路

我们的目标是:不改动原有Pytest脚本逻辑,仅通过"封装+接口"的方式,让脚本项目具备平台接入能力

实现原则:

  1. 不重写:保持原有 testcases/utils/ 目录结构不变
  2. 不侵入:原有脚本文件无需修改任何代码
  3. 不解耦:仅新增API层,不改变原有执行逻辑

2. 目录改造

原有脚本项目核心目录保留不变,改造后新增「API 层、配置层」,确保原有脚本无需修改即可复用。

  • 改造前目录(即脚本项目原始目录)示例如下:

    SUPER-API-AUTO-TEST/          # 接口自动化测试项目根目录
    ├── auth.py                    # 鉴权相关
    ├── case_generator.py          # 用例生成逻辑
    ├── config.yaml                # 项目配置
    ├── conftest.py                # Pytest夹具
    ├── runner.py                  # 用例执行入口
    ├── reports/                   # 测试报告目录
    ├── logs/                      # 日志目录
    ├── testcases/                 # 测试用例脚本(Python)
    ├── testcases_data/            # 测试用例数据(YAML)
    └── utils/                     # 通用工具类
    
  • Fastapi 改造后目录结构示例如下:

    FASTAPI-SUPER-API-AUTO-TEST/  # 基于FastAPI的改造后目录
    ├── auth.py                    
    ├── case_generator.py          
    ├── conftest.py                
    ├── main.py                   # FastAPI应用入口(核心新增)
    ├── pytest.ini                
    ├── runner.py                  
    ├── api/                      # API层(核心新增)
    │   ├── testcase_route.py     # 接口路由定义(URL、请求参数)
    │   ├── testcase_service.py   # 接口业务逻辑(与原有脚本交互)
    │   └── __init__.py           
    ├── configs/                  # 配置目录(拆分原有config.yaml)
    │   └── config.yaml           
    ├── logs/                     
    ├── reports/                  
    ├── testcases/                # 完全复用原有目录
    ├── testcases_data/           # 完全复用原有目录
    └── utils/                    # 完全复用原有目录
    

改造后,新增了 Fastapi 路由层 api/、配置层 configs/,以及 FastAPI 应用入口 main.py

三. 核心接口设计与实现

1. 接口设计示例

我们设计了以下关键接口供测试平台调用:

① 获取项目与模块信息

  • GET /api_test/testcases/projects:获取所有项目列表
  • GET /api_test/testcases/modules:获取指定项目下的模块列表

② 获取用例列表

  • GET /api_test/testcases/list:支持按项目/模块筛选测试用例

③ 生成测试用例

  • POST /api_test/testcases/generate:根据YAML用例文件生成Python测试脚本

④ 执行测试任务

  • POST /api_test/testcases/run:后台执行指定测试用例,支持环境选择、报告类型、回调通知等

⑤ 获取测试报告

  • GET /api_test/reports/get_by_task:根据任务ID获取报告访问地址

2. 代码示例

testcase_service.py

# @author:  xiaoqq

import os, re
from datetime import datetime
from typing import List, Optional, Dict
from pathlib import Path

TESTCASE_ROOT = "testcases"

def get_abs_root_path(root_path: str) -> Path:
    """
    使用当前文件相对路径构造 testcases/ 的绝对路径
    :param root_path: 目录名
    :return:
    """
    base_dir = Path(__file__).resolve().parent  # 当前文件所在目录
    abs_root_path = (base_dir.parent / root_path).resolve()
    return abs_root_path

def get_all_testcases(project: Optional[str] = None,
                  module: Optional[str] = None,
                  root_path: str = TESTCASE_ROOT) -> List[Dict]:
    """
    获取所有测试用例(支持通过 project/module 筛选)
    返回字段包括 filename(无后缀)、path(绝对路径字符串)、Allure 元信息等
    """
    abs_root_path = get_abs_root_path(root_path)
    if not abs_root_path.exists():
       return []
    
    # 路径校验
    if module and not project:
       raise ValueError("传入 module 前必须先传入 project")
    
    # 构造起始目录路径
    search_path = abs_root_path
    if project:
       search_path = search_path / project
    if module:
       search_path = search_path / module
    
    if not search_path.exists():
       return []
    
    testcases = []
    
    for dirpath, _, filenames in os.walk(search_path):
       for file in filenames:
          if file.startswith("test_") and file.endswith(".py"):
             full_path = os.path.join(dirpath, file)
             rel_path = os.path.relpath(full_path, abs_root_path)  # 相对路径,如 merchant/device/test_xxx.py
             path_parts = Path(rel_path).parts  # 使用 pathlib 安全拆解路径
             
             if len(path_parts ) < 2:
                continue  # 至少要有 project/filename 结构
             
             _project = path_parts [0]
             _filename = path_parts [-1]
             _module = path_parts [1] if len(path_parts ) > 2 else None  # module 可选
             
             # 按传参过滤
             if project and _project != project:
                continue
             if module and _module != module:
                continue
             
             filename = os.path.splitext(_filename)[0]  # 去掉 .py 后缀
             last_modified = datetime.fromtimestamp(os.path.getmtime(full_path)).isoformat()
             
             # 提取用例元信息
             try:
                case_name, epic, feature, story = extract_case_info(full_path)
             except Exception as e:
                case_name, epic, feature, story = None, None, None, None
                
             # 拼接最终 path 字段为 TESTCASE_ROOT/... 形式
             full_case_path = str(Path(root_path) / rel_path).replace("\\", "/")
             
             # 构造 external_id:project|module|filename|path
             external_id = f"{_project}|{_module or 'nomodule'}|{filename}|{full_case_path}"
             
             testcases.append({
                "project": _project,
                "module": _module,  # None 表示无 module 层级
                "file": _filename,
                "filename": filename,
                "path": full_case_path,
                "last_modified": last_modified,
                "case_name": case_name or filename,
                "allure_epic": epic,
                "allure_feature": feature,
                "allure_story": story,
                "external_id": external_id # 加入唯一标识
             })
    
    return testcases

def extract_case_info(file_path):
    """
    解析测试用例文件,获取相应信息
    :param file_path:
    :return:
    """
    with open(file_path, 'r', encoding='utf-8') as file:
       content = file.read()
       
       case_name_match = re.search(
          r'def setup_class.*?\(.*?\):.*?log\.info\(\'========== 开始执行测试用例:(.+?) ==========\'',
          content, re.DOTALL
       )
       case_name = case_name_match.group(1).strip() if case_name_match else \
       os.path.splitext(os.path.basename(file_path))[0]
       
       allure_epic_match = re.search(r'@allure\.epic\(\'(.+?)\'\)', content)
       allure_feature_match = re.search(r'@allure\.feature\(\'(.+?)\'\)', content)
       allure_story_match = re.search(r'@allure\.story\(\'(.+?)\'\)', content)
       
       allure_epic = allure_epic_match.group(1).strip() if allure_epic_match else None
       allure_feature = allure_feature_match.group(1).strip() if allure_feature_match else None
       allure_story = allure_story_match.group(1).strip() if allure_story_match else None
       
       return case_name, allure_epic, allure_feature, allure_story



def get_all_projects(root_path: str = TESTCASE_ROOT) -> List[Dict[str, str]]:
    """
    获取 testcases/ 下所有项目名、相对路径及创建时间(倒序排序)
    """
    abs_root_path = get_abs_root_path(root_path)
    if not abs_root_path.exists():
       return []
    
    projects = []
    for d in abs_root_path.iterdir():
       if d.is_dir():
          created_time = datetime.fromtimestamp(d.stat().st_ctime)
          projects.append({
             "name": d.name,
             "path": str(Path(root_path) / d.name).replace("\\", "/"),
             "created_time": created_time.isoformat()
          })
    
    # 按创建时间倒序
    return sorted(projects, key=lambda x: x["created_time"], reverse=True)


def get_all_projects_and_modules(
    project: Optional[str] = None,
    root_path: str = TESTCASE_ROOT
) -> List[Dict]:
    """
    获取所有项目和模块结构(支持指定项目)。包含路径、创建时间,按项目时间倒序。
    """
    abs_root_path = get_abs_root_path(root_path)
    if not abs_root_path.exists():
       return []
    
    result = []
    
    for proj_dir in abs_root_path.iterdir():
       if not proj_dir.is_dir():
          continue
       
       proj_name = proj_dir.name
       if project and proj_name != project:
          continue
       
       proj_created_time = datetime.fromtimestamp(proj_dir.stat().st_ctime)
       modules = []
       
       # 遍历模块目录时需要忽略的子目录
       EXCLUDE_DIRS = {"__pycache__", ".pytest_cache", ".git", ".idea"}
       for mod_dir in proj_dir.iterdir():
          if mod_dir.is_dir() and mod_dir.name not in EXCLUDE_DIRS:
             mod_created_time = datetime.fromtimestamp(mod_dir.stat().st_ctime)
             modules.append({
                "name": mod_dir.name,
                "path": str(Path(root_path) / proj_name / mod_dir.name).replace("\\", "/"),
                "created_time": mod_created_time.isoformat()
             })
       
       # 模块也可以排序(如有需求)
       modules.sort(key=lambda x: x["created_time"], reverse=True)
       
       result.append({
          "project": proj_name,
          "path": str(Path(root_path) / proj_name).replace("\\", "/"),
          "created_time": proj_created_time.isoformat(),
          "modules": modules
       })
       
       if project:
          break
    
    # 项目排序
    return sorted(result, key=lambda x: x["created_time"], reverse=True)


def generate_testcase(case_yaml_list: list = None):
    """
    生成测试用例
    :return:
    """
    from case_generator import CaseGenerator
    CG = CaseGenerator()
    CG.generate_testcases(project_yaml_list=case_yaml_list)


if __name__ == '__main__':
    # print(get_all_testcases())
    # print(get_all_projects())
    print(get_all_projects_and_modules(project="merchant"))

testcase_route.py 示例如下:

# @author:  xiaoqq

from pathlib import Path
from fastapi import APIRouter, BackgroundTasks, Query, Body
from pydantic import BaseModel
from typing import List, Optional
from runner import run_tests
from api.testcase_service import (
	get_all_testcases,
	get_all_projects,
	get_all_projects_and_modules,
	generate_testcase,
)

router = APIRouter()

class TestExecutionRequest(BaseModel):
	testcases: Optional[List[str]] = ['testcases/']  # 默认运行所有目录
	env: Optional[str] = 'pre'
	report_type: Optional[str] = 'pytest-html'
	dingtalk_notify: Optional[bool] = True
	task_id: Optional[str]
	callback_url: Optional[str]
	auth_token: Optional[str] = None  # 新增字段:从平台传入的 token

# 执行测试用例
@router.post("/testcases/run")
def run_testcases(request: TestExecutionRequest, background_tasks: BackgroundTasks):
	try:
		background_tasks.add_task(
			run_tests,
			testcases=request.testcases,
			env=request.env,
			report_type=request.report_type,
			dingtalk_notify=request.dingtalk_notify,
			task_id=request.task_id,
			callback_url=request.callback_url,
			auth_token=request.auth_token,  # 测试平台回调 auth_token
		)
		return {
			"code": 0,
			"msg": "测试任务已提交后台执行",
			"task_id": request.task_id
		}
	except Exception as e:
		return {"code":1, "msg": f"测试任务失败:{str(e)}"}

# 获取测试用例
@router.get("/testcases/list")
def list_testcases(project: str = Query(None), module: str = Query(None)):
	try:
		testcases = get_all_testcases(project, module)
		return {
			"code": 0,
			"msg": "success",
			"testcases": testcases
		}
	except Exception as e:
		return {"code": 1, "msg": f"获取测试用例失败:{str(e)}"}


# 获取 testcases/ 中的所有测试项目
@router.get("/testcases/projects")
def list_projects():
	try:
		projects = get_all_projects()
		return {"code": 0, "msg": "success", "projects": projects}
	except Exception as e:
		return {"code": 1, "msg": f"获取测试项目失败:{str(e)}"}
	

# 获取 testcases/ 中的所有测试项目及模块
@router.get("/testcases/modules")
def list_modules(project: str = Query(None)):
	try:
		modules = get_all_projects_and_modules(project)
		return {"code": 0, "msg": "success", "modules": modules}
	except Exception as e:
		return {"code": 1, "msg": f"获取测试项目-模块失败:{str(e)}"}



class GenerateCaseRequest(BaseModel):
	case_yaml_list: Optional[List[str]] = None

# 根据 testcases_data/ 中的测试数据生成测试用例文件
@router.post("/testcases/generate")
def generate_testcase_route(req: GenerateCaseRequest):
	try:
		generate_testcase(req.case_yaml_list)
		return {"code": 0, "msg": "success"}
	except Exception as e:
		return {"code": 1, "msg": f"获取测试项目-模块失败:{str(e)}"}


@router.get("/reports/get_by_task")
def get_report_by_task(
		task_id: str,
		report_type: str,
		created_at: str  # 格式: "20250814"
):
	"""
	根据 task_id + 创建时间 + report_type 获取报告 URL
	"""
	if not created_at:
		return {"code": 1, "msg": "created_at 必填", "url": None}
	
	base_path = Path(__file__).resolve().parent.parent / "reports" / created_at
	
	if report_type == "pytest-html":
		report_file = base_path / f"report_{task_id}.html"
	elif report_type == "allure":
		report_file = base_path / f"report_{task_id}_allure/html/index.html"
	else:
		return {"code": 1, "msg": "未知 report_type", "url": None}
	
	if not report_file.exists():
		return {"code": 1, "msg": "报告文件不存在", "url": None}
	
	relative_url = str(report_file.relative_to(Path(__file__).resolve().parent.parent)).replace("\\", "/")
	return {"code": 0, "msg": "success", "url": f"/{relative_url}"}

mian.py

from fastapi import FastAPI
from api import testcase_route
from pathlib import Path
from fastapi.staticfiles import StaticFiles

app = FastAPI(title="接口自动化测试服务")

# 挂载测试用例路由
app.include_router(testcase_route.router, prefix="/api_test", tags=["测试任务"])

# 挂载 reports 目录为静态文件目录
reports_dir = Path(__file__).parent / "reports"
reports_dir.mkdir(exist_ok=True)  # 确保目录存在
app.mount("/reports", StaticFiles(directory=reports_dir), name="reports")

if __name__ == "__main__":
    from utils.log_manager import LogManager
    LogManager.setup_logging()  # 启动时显式初始化日志
    
    import uvicorn
    uvicorn.run(
        "main:app",
        host="0.0.0.0",
        port=8000,
        # reload=True,
        reload_excludes=["testcases/*", "logs/*", "reports/*"]  # 排除这些目录的文件变更
    )

四. 测试平台调用

执行mian.py,启动 Fastapi 项目后,便可在测试平台通过调用相关接口来管理该脚本测试项目(平台调用代码不具体提供)。

1. 调用示意图

测试平台
   │
   │ HTTP 调用
   ▼
FastAPI 测试服务
   │
   │ pytest 执行
   ▼
测试报告生成
   │
   │ 回调结果
   ▼
测试平台展示

这样,职责边界非常清晰:

  • 测试平台:调度、记录、展示,
  • 改造后的测试服务:执行、产出报告

2. 测试平台界面

平台测试用例列表:

image-20260120134215814

测试报告列表:

image-20260122143715247

五. 总结

方案优势总结如下:

  1. 解耦与复用:脚本项目独立维护,平台通过接口调用,互不影响

  2. 灵活执行:支持按项目、模块、用例筛选执行,适应不同测试场景。

  3. 异步处理:长时间任务后台执行,平台可实时获取状态与报告。

  4. 报告统一管理:所有报告集中存储,支持在线统一查看。

当然,示例代码还可以进行优化扩展,如加入用户认证机制来保障接口安全等。

当接口自动化发展到一定规模,单机脚本Jenkins Job 都会成为瓶颈,而“脚本服务化 + 平台调度”,几乎是所有成熟团队最终都会走到的一步。

如果你:

  • 正在做接口自动化
  • 或正在参与测试平台建设
  • 或正在被“脚本怎么接平台”折磨

那么,希望这篇文章能少让你走一点弯路。

posted @ 2026-01-25 14:46  给你一页白纸  阅读(16)  评论(2)    收藏  举报