39. 自动化异步测试开发之编写异步业务函数、测试函数和测试类(函数写法)

39. 自动化异步测试开发之编写异步业务函数、测试函数和测试类(函数写法)

一、异步业务函数解析

1.1 页面导航函数

async def get(async_driver, url: str = 'http://secure.smartbearsoftware.com/samples/testcomplete12/WebOrders/Login.aspx'):await async_driver.get(url)
  • 功能:导航到指定URL
  • 默认URL:Web Orders登录页面
  • 执行效果:浏览器打开指定页面

1.2 登录功能函数

async def login(async_driver, username: str = 'Tester', password: str = 'test'):await async_driver.send_keys('id', 'ctl00_MainContent_username', text=username)await async_driver.send_keys('id', 'ctl00_MainContent_password', text=password)await async_driver.click('name', 'ctl00$MainContent$login_button')
  • 操作步骤
    1. 在用户名输入框输入用户名
    2. 在密码输入框输入密码
    3. 点击登录按钮
  • 默认凭证:用户名’Tester’,密码’test’
  • 元素定位:使用ID定位输入框,Name定位按钮

1.3 搜索功能函数

async def search(async_driver):await async_driver.click('xpath', '//*[@id="ctl00_menu"]/li[3]/a')  # 点击搜索菜单await async_driver.send_keys('id', 'ctl00_MainContent_fmwOrder_txtName')  # 输入搜索内容await async_driver.click('id', 'ctl00_MainContent_fmwOrder_InsertButton')  # 点击搜索按钮
  • 操作流程
    1. 点击菜单中的搜索选项
    2. 在搜索框输入内容
    3. 点击搜索按钮

1.4 登出功能函数

async def logout(async_driver):await async_driver.click('xpath', '//*[@id="ctl00_logout"]')  # 点击登出链接
  • 功能:退出当前登录状态
  • 定位方式:使用XPath定位登出链接

二、异步测试函数实现

2.1 登录功能测试

async def test_login(async_driver):await get(async_driver)  # 打开登录页面await login(async_driver)  # 执行登录操作# 验证登录成功title_text = await async_driver.text('xpath', '//*[@id="aspnetForm"]//td[1]/h1')assert title_text == 'Web Orders'await logout(async_driver)  # 退出登录
  • 测试流程
    1. 打开登录页
    2. 输入凭证登录
    3. 验证页面标题
    4. 登出系统
  • 断言验证:检查登录后页面标题是否为’Web Orders’

2.2 搜索功能测试

async def test_search(async_driver):await get(async_driver)  # 打开登录页await login(async_driver)  # 登录系统await search(async_driver)  # 执行搜索操作# 验证错误提示error_msg = await async_driver.text('id', "ctl00_MainContent_fmwOrder_RequiredFieldValidator3")assert error_msg == "Field 'Street' cannot be empty."await logout(async_driver)  # 退出登录
  • 测试流程
    1. 登录系统
    2. 执行空搜索
    3. 验证错误提示
  • 断言验证:检查是否显示’Street不能为空’的错误提示

三、完整测试执行流程

3.1 测试运行器实现

import asyncio
from chap9.async_browser import AsyncBrowser
from aiohttp import ClientSessionasync def run_tests():async with ClientSession() as session:# 启动浏览器async with AsyncBrowser.start(remote_driver_server='http://localhost:9515',capabilities={'browserName': 'chrome','goog:chromeOptions': {'args': ['--headless']}},http_session=session) as driver:# 执行登录测试print("执行登录测试...")await test_login(driver)print("登录测试通过 ✓")# 执行搜索测试print("执行搜索测试...")await test_search(driver)print("搜索测试通过 ✓")if __name__ == "__main__":asyncio.run(run_tests())

3.2 预期执行结果

执行登录测试...
登录测试通过 ✓
执行搜索测试...
搜索测试通过 ✓

3.3 实际页面操作流程

1. 打开登录页:http://secure.smartbearsoftware.com/...
2. 输入用户名:Tester
3. 输入密码:test
4. 点击登录按钮
5. 验证页面标题:Web Orders
6. 点击登出链接
7. 重新登录
8. 点击搜索菜单
9. 点击搜索按钮(不输入内容)
10. 验证错误提示:Field 'Street' cannot be empty.
11. 点击登出链接

四、设计优势分析

4.1 业务与测试分离

  • 业务函数:封装页面操作逻辑(如login, search)
  • 测试函数:组合业务函数并添加断言
  • 分离好处:业务变更只需修改一处

4.2 异步执行优势

操作同步执行时间异步执行时间提升效果
打开页面2秒0.5秒75%
输入操作1秒0.3秒70%
多测试并行线性增长并行执行300%+

4.3 可重用性设计

# 在不同测试中重用业务函数
async def test_order(async_driver):await get(async_driver)await login(async_driver)# 添加订单测试代码await logout(async_driver)

五、最佳实践建议

  1. 参数化默认值

    async def login(async_driver, username: str = DEFAULT_USER, password: str = DEFAULT_PASS):
    
  2. 元素定位器集中管理

    USERNAME_FIELD = ('id', 'ctl00_MainContent_username')
    await async_driver.send_keys(*USERNAME_FIELD, text=username)
    
  3. 添加操作等待

    from selenium.webdriver.support.ui import WebDriverWait
    await WebDriverWait(async_driver, 10).until(element_visible(USERNAME_FIELD))
    
  4. 错误处理增强

    async def safe_login(async_driver):try:await login(async_driver)except LoginException:await handle_login_failure()
    

这种异步测试开发模式通过将业务操作、测试验证和测试执行分层设计,显著提高了测试代码的可维护性和执行效率。

六、完整代码

"""
Python :3.13.3
Selenium: 4.31.0async_test_func.py
"""async def get(async_driver,url: str = 'http://secure.smartbearsoftware.com/samples/testcomplete12/WebOrders/Login.aspx'):await async_driver.get(url)async def login(async_driver, username: str = 'Tester', password: str = 'test'):await async_driver.send_keys('id', 'ctl00_MainContent_username', text=username)await async_driver.send_keys('id', 'ctl00_MainContent_password', text=password)await async_driver.click('name', 'ctl00$MainContent$login_button')async def search(async_driver):await async_driver.click('xpath', '//*[@id="ctl00_menu"]/li[3]/a')await async_driver.send_keys('id', 'ctl00_MainContent_fmwOrder_txtName')await async_driver.click('id', 'ctl00_MainContent_fmwOrder_InsertButton')async def logout(async_driver):await async_driver.click('xpath', '//*[@id="ctl00_logout"]')async def test_login(async_driver):await get(async_driver)await login(async_driver)assert await async_driver.text('xpath', '//*[@id="aspnetForm"]//td[1]/h1') == 'Web Orders'await logout(async_driver)async def test_search(async_driver):await get(async_driver)await login(async_driver)await search(async_driver)assert await async_driver.text('id',"ctl00_MainContent_fmwOrder_RequiredFieldValidator3") == "Field 'Street' cannot be empty."await logout(async_driver)

「小贴士」:点击头像→【关注】按钮,获取更多软件测试的晋升认知不迷路! 🚀

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/bicheng/83451.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Qt 无边框窗口实现拖动与窗口控制(最小化/最大化/关闭)

在 Qt 中,使用 Qt::FramelessWindowHint 可以创建无边框窗口,但这样会导致窗口无法拖动,并且系统默认的标题栏按钮(最小化、最大化、关闭)也会消失。本文将介绍如何实现无边框窗口的鼠标拖动功能,并添加自定…

Linux中的System V通信标准-共享内存、消息队列以及信号量

在Linux系统中,System V IPC(Inter-Process Communication)提供了一系列进程间通信的机制,包括共享内存、消息队列和信号量。这些机制在系统中发挥了重要作用,帮助进程之间进行数据交换和同步。本文将详细介绍这些机制…

postman工具使用

基本功能操作 常用断言 定义:postman 断言借助 JavaScript - js 语言编写代码,自动判断预期结果与实际结果是否一致。( 注意断言 代码写在 Tests 的标签中) 断言响应状态码 断言响应体是否包含某个字符串(Response bo…

VBA数据库解决方案二十:Select表达式From区域Where条件Order by

《VBA数据库解决方案》教程(版权10090845)是我推出的第二套教程,目前已经是第二版修订了。这套教程定位于中级,是学完字典后的另一个专题讲解。数据库是数据处理的利器,教程中详细介绍了利用ADO连接ACCDB和EXCEL的方法…

算法-集合的使用

1、set常用操作 set<int> q; //以int型为例 默认按键值升序 set<int,greater<int>> p; //降序排列 int x; q.insert(x); //将x插入q中 q.erase(x); //删除q中的x元素,返回0或1,0表示set中不存在x q.clear(); //清空q q.empty(); //判断q是否为空&a…

C++文件和流基础

C文件和流基础 1. C文件和流基础1.1 文件和流的概念1.2 标准库支持1.3 常用文件流类ifstream 类ofstream 类fstream 类 2.1 打开文件使用构造函数打开文件使用 open() 成员函数打开文件打开文件的模式标志 2.2 关闭文件使用 close() 成员函数关闭文件关闭文件的重要性 3.1 写入…

Maven---配置本地仓库

目录 5. 5.1在Maven路径下新建文件夹用于本地仓库存储 5.2 复制本地仓库路径 5.3 找到配置文件路径&#xff0c;使用VSCode方式打开 5.4 新增一行代码 5.5 复制本地仓库路径&#xff0c;设置存储路径 5.1在Maven路径下新建文件夹用于本地仓库存储 5.2 复制本地仓库路径 5…

Vue3 + Element Plus + TypeScript 中 el-cascader 实现模拟用户点击功能

模拟点击&#xff0c;调用 el-cascader 的公开方法 togglePopperVisible 来展开下拉框 MaterialOut.vue <script setup lang"ts" name"MaterialOut"> ...... import { ElMessage, type ElCascader } from "element-plus";// 级联组件实例…

新能源汽车与油车销量

中国油车与新能源车销量对比&#xff08;2022-2025年&#xff09; ‌1. 市场份额演化&#xff08;2022-2025年&#xff09;‌ ‌年份‌ ‌新能源车销量 &#xff08;渗透率&#xff09;‌ ‌燃油车销量 &#xff08;渗透率&#xff09;‌ ‌关键事件‌ ‌2022‌ 688.7万辆…

C++ list代码练习、set基础概念、set对象创建、set大小操作

对应力扣&#xff0c;回文链表&#xff0c;代码见下 /*** Definition for singly-linked list.* struct ListNode {* int val;* ListNode *next;* ListNode() : val(0), next(nullptr) {}* ListNode(int x) : val(x), next(nullptr) {}* ListNode(int x, …

前端面试宝典---前端水印

明水印 1. 背景图 通过css的background-image加载背景图 2. canvasbackground水印 前端水印实现思路与示例代码 一、核心实现思路 Canvas动态生成水印 通过Canvas绘制文本或图案&#xff0c;将生成的图像转为Base64格式&#xff0c;作为背景图重复平铺到目标元素上。例如&…

恶意软件清理工具,让Mac电脑安全更简单

​你的Mac最近是不是开始表演"电子迷惑行为"&#xff1f;浏览器主页突然变成澳门赌场&#xff0c;风扇转得比直升机螺旋桨还猛......恭喜你&#xff01;可能中奖获得"恶意软件大礼包"&#xff01;别慌&#xff0c;今天就教你用恶意软件清理工具化身数字特工…

Spring Boot 3.X 下Redis缓存的尝试(二):自动注解实现自动化缓存操作

前言 上文我们做了在Spring Boot下对Redis的基本操作&#xff0c;如果频繁对Redis进行操作而写对应的方法显示使用注释更会更高效&#xff1b; 比如&#xff1a; 依之前操作对一个业务进行定入缓存需要把数据拉取到后再定入&#xff1b; 而今天我们可以通过注释的方式不需要额外…

Deepseek应用技巧-Dify安装和踩坑指南

前言&#xff1a;Dify的名号是非常大的&#xff0c;作为私有化AI部署中必不可少的一个组件&#xff0c;他的功能和COZE十分相似&#xff0c;可以进行工作流和智能体的搭建&#xff0c;有非常强大的功能&#xff0c;那本节就将来揭开Dify的神秘的面纱&#xff0c;首先看一下Dify…

ubuntu24.04安装教程(图文详解)

Ubuntu 24.04 LTS&#xff0c;代号 Noble Numbat&#xff0c;于 2024 年 4 月 25 日发布&#xff0c;现在可以从 Ubuntu 官方网站及其镜像下载。此版本将在 2029 年 4 月之前接收为期五年的官方安全和维护更新。 关于 Ubuntu 24.04 LTS 的一些关键点&#xff1a; 发布日期&am…

数据绑定页面的完整的原理、逻辑关系、实现路径是什么?页面、表格、字段、属性、值、按钮、事件、模型、脚本、服务编排、连接器等之间的关系又是什么?

目录 一、核心概念:什么是数据绑定页面? 二、涉及的组件及其逻辑关系 页面(Page): 表格(Table): 字段(Field): 属性(Property): 值(Value): 按钮(Button): 事件(Event): 模型(Model): 脚本(Script): 服务(Service): 服务编排(Se…

【 SpringCloud | 微服务 网关技术 】

单体架构时我们只需要完成一次用户登录、身份校验&#xff0c;就可以在所有业务中获取到用户信息。而微服务拆分后&#xff0c;每个微服务都独立部署&#xff0c;这就存在一些问题&#xff1a; 每个微服务都需要编写登录校验、用户信息获取的功能吗&#xff1f; 当微服务之间调…

python,Dataframe基于所有包含某个关键字的列等于某个值过滤

在 Python 中&#xff0c;使用 Pandas 的 DataFrame 丢弃符合特定条件的行&#xff0c;条件为所有包含某个关键字的列中&#xff0c;等于某个值&#xff08;即所有包含某个关键字的列中等于某个值的行&#xff09;&#xff0c;可用以下方法实现&#xff1a; import pandas as …

50天50个小项目 (Vue3 + Tailwindcss V4) ✨ | Sound Board(音响控制面板)

&#x1f4c5; 我们继续 50 个小项目挑战&#xff01;—— SoundBoard 组件 仓库地址&#xff1a;https://github.com/SunACong/50-vue-projects 项目预览地址&#xff1a;https://50-vue-projects.vercel.app/ &#x1f3af; 组件目标 实现一个响应式按钮面板&#xff0c;点…

在Ubuntu20.04上安装ROS Noetic

本章教程,主要记录在Ubuntu20.04上安装ROS Noetic。 一、添加软件源 sudo sh -c . /etc/lsb-release && echo "deb http://mirrors.tuna.tsinghua.edu.cn/ros/ubuntu/ `lsb_release -cs` main" > /etc/apt/sources.list.d/ros-latest.list二、设置秘钥 …