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

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

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

一、类结构设计解析

1.1 基类设计

class Base:async_driver = None  # 🚗 存储浏览器驱动实例async def get(self, url: str = 'http://secure.smartbearsoftware.com/...'):await self.async_driver.get(url)  # 🌐 导航到指定URL
  • 核心成员async_driver 存储浏览器驱动实例
  • 通用功能:提供页面导航方法
  • 默认URL:Web Orders登录页面

1.2 登录页面类

class LoginPage(Base):async def login(self, username: str = 'Tester', password: str = 'test'):# 输入用户名await self.async_driver.send_keys('id', 'ctl00_MainContent_username', text=username)# 输入密码await self.async_driver.send_keys('id', 'ctl00_MainContent_password', text=password)# 点击登录按钮await self.async_driver.click('name', 'ctl00$MainContent$login_button')
  • 功能定位:封装登录相关操作
  • 默认凭证:用户名’Tester’,密码’test’
  • 元素定位:使用ID定位输入框,Name定位按钮

1.3 主页面类

class MainPage(LoginPage):async def search(self):# 点击搜索菜单await self.async_driver.click('xpath', '//*[@id="ctl00_menu"]/li[3]/a')# 输入搜索内容(需要添加text参数)await self.async_driver.send_keys('id', 'ctl00_MainContent_fmwOrder_txtName')# 点击搜索按钮await self.async_driver.click('id', 'ctl00_MainContent_fmwOrder_InsertButton')async def logout(self):# 点击登出链接await self.async_driver.click('xpath', '//*[@id="ctl00_logout"]')
  • 继承关系:继承自LoginPage,复用登录功能
  • 扩展功能:添加搜索和登出操作
  • 元素定位:混合使用XPath和ID定位器

二、测试类实现

2.1 登录测试类

class AsyncTestLogin(MainPage):async def test_login(self, *args):await self.get()  # 打开登录页await self.login(*args)  # 执行登录# 验证登录成功title_text = await self.async_driver.text('xpath', '//*[@id="aspnetForm"]//td[1]/h1')assert title_text == 'Web Orders'
  • 测试流程
    1. 打开登录页
    2. 执行登录
    3. 验证页面标题
  • 断言验证:检查标题是否为’Web Orders’

2.2 搜索测试类

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

三、完整测试执行示例

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'},http_session=session) as driver:# 创建测试实例login_test = AsyncTestLogin()login_test.async_driver = driversearch_test = AsyncTestMain()search_test.async_driver = driver# 执行登录测试print("执行登录测试...")await login_test.test_login()print("登录测试通过 ✓")# 执行搜索测试print("执行搜索测试...")await search_test.test_search()print("搜索测试通过 ✓")if __name__ == "__main__":asyncio.run(run_tests())

3.2 预期执行结果

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

3.3 实际操作流程

登录测试:1. 打开登录页2. 输入用户名:Tester3. 输入密码:test4. 点击登录按钮5. 验证页面标题:Web Orders搜索测试:1. 打开登录页2. 输入凭证登录3. 点击搜索菜单4. 点击搜索按钮(不输入内容)5. 验证错误提示:Field 'Street' cannot be empty.6. 点击登出链接

四、类写法的优势分析

4.1 继承结构

Base│├── LoginPage│     ││     └── MainPage│           ││           ├── AsyncTestLogin│           ││           └── AsyncTestMain
  • 代码复用:通过继承复用公共方法和属性
  • 功能扩展:子类可以扩展或重写父类方法
  • 逻辑分层:清晰区分页面操作和测试验证

4.2 与函数写法的对比

特性类写法函数写法
状态管理通过类属性维护状态依赖参数传递状态
代码组织按页面/功能模块组织按操作流程组织
复用性高(继承机制)中(函数组合)
学习曲线较陡峭(需理解OOP)较平缓
适用场景大型项目/复杂页面小型项目/简单流程

五、最佳实践建议

5.1 类设计优化

class BasePage:def __init__(self, driver):self.driver = driver  # ✅ 通过构造器注入驱动async def open(self, url):await self.driver.get(url)class LoginPage(BasePage):USERNAME = ('id', 'ctl00_MainContent_username')PASSWORD = ('id', 'ctl00_MainContent_password')LOGIN_BTN = ('name', 'ctl00$MainContent$login_button')async def login(self, username, password):await self.driver.send_keys(*self.USERNAME, text=username)await self.driver.send_keys(*self.PASSWORD, text=password)await self.driver.click(*self.LOGIN_BTN)

5.2 测试类优化

class TestLogin(LoginPage):TITLE = ('xpath', '//*[@id="aspnetForm"]//td[1]/h1')async def test_success_login(self):await self.open(LOGIN_URL)await self.login(TEST_USER, TEST_PASS)assert await self.driver.text(*self.TITLE) == 'Web Orders'

5.3 执行入口优化

async def main():driver = await create_driver()login_page = LoginPage(driver)await login_page.test_success_login()

这种基于类的异步测试开发模式,通过面向对象的设计思想,提供了更结构化、可维护性更高的测试代码组织方式,特别适合中大型自动化测试项目。

六、完整代码

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

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

http://www.xdnf.cn/news/730009.html

相关文章:

  • Delphi读写Mifare Desfire Ev1 Ev2 EV3卡修改AES密钥源码
  • 39. 自动化异步测试开发之编写异步业务函数、测试函数和测试类(函数写法)
  • 逆向分析基础总结
  • HTML 文件反编译指南:优化与学习网页代码
  • 【容器docker】启动容器kibana报错:“message“:“Error: Cannot find module ‘./logs‘
  • STUSB4500 PPS(PD3.0)快充SINK模块——应用 解析
  • [学习] C语言的回调函数(代码示例)
  • 数据基座觉醒!大数据+AI如何重构企业智能决策金字塔(下)
  • 【Linux 学习计划】-- 命令行参数 | 环境变量
  • 【目标检测】【AAAI-2022】Anchor DETR
  • 【Golang进阶】第八章:并发编程基础——从Goroutine调度到Channel通信实战
  • Redis持久化机制
  • MPC5744P——eTimer简介
  • Github 2025-05-30Java开源项目日报Top10
  • 《深入解析Go语言结构:简洁高效的工程化设计》
  • 基于 KubeKey 3.1.9,快速部署 K8s 1.33.0 高可用集群
  • Java复习Day23
  • haproxy 搭建web群集
  • EMQX社区版5.8.5集群搭建踩坑记
  • vscode命令行debug
  • 中国外卖包装废弃物高精度网格图谱(Tif/Excel/Shp)
  • 128、STM32H723ZGT6实现串口IAP
  • 贪心算法实战3
  • 6年“豹变”,vivo S30系列引领手机进入场景“体验定义”时代
  • 交叉编译tcpdump工具
  • File—IO流
  • Vue 3.0 中的路由导航守卫详解
  • [yolov11改进系列]基于yolov11引入轻量级注意力机制模块ECA的python源码+训练源码
  • CVPR 2025论文分享|MGGTalk:一个更加通用的说话头像动画生成框架
  • 60天python训练计划----day40