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

19. 结合Selenium和YAML对页面实例化PO对象改造

19. 结合Selenium和YAML对页面实例化PO对象改造

一、架构升级核心思路

1.1 改造核心目标

# 原始PO模式:显式定义元素定位
username = ('id', 'ctl00_MainContent_username')# 改造后PO模式:动态属性访问
self.username.send_keys('Tester')  # 自动触发元素定位

1.2 关键技术实现

  • 元编程技术:通过__getattr__实现动态属性访问
  • 配置驱动模式:YAML文件存储元素定位策略
  • 链式继承体系:实现跨页面元素复用

二、核心类改造解析

2.1 页面基类增强

class Page:locators = {}  # 元素定位池browser = CHROME  # 浏览器类型绑定def __getattr__(self, loc):"""动态属性访问拦截器"""if loc not in self.locators:raise AttributeError(f"'{self.__class__.__name__}'未定义元素'{loc}'")by, val = self.locators[loc]  # 解构定位策略return self.driver.find_element(by, val)  # 延迟定位执行
核心机制:
  • 按需定位:元素首次访问时执行定位
  • 异常封装:自动抛出可读性错误
  • 驱动管理:统一浏览器实例生命周期

三、配置管理系统升级

3.1 setting.py核心配置

# YAML元素配置文件映射
YAML_ELEMENT = {'cp': join(ELEMENTS_PATH, 'CommonLoginPass.yml'),'op': join(ELEMENTS_PATH, 'oder_page.yml')
}# 浏览器启动参数
CHROME_EXP = {'excludeSwitches': ['enable-automation'],'mobileEmulation': {'deviceName': 'iPhone 12'}
}

3.2 配置加载方式

class CommonLoginPage(Page):locators = YamlReader(YAML_ELEMENT['cp']).data  # 动态加载登录页配置class MainPage(CommonLoginPage):locators.update(YamlReader(YAML_ELEMENT['op']).data)  # 继承并扩展配置

四、页面类实现模式

4.1 登录页面实现

class CommonLoginPage(Page):url = PROJECT_Oder_URLdef login(self, username='Tester'):self.driver.get(self.url)self.username.send_keys(username)  # 动态属性访问self.password.send_keys('test')self.loginBtn.click()

4.2 主页面扩展

class MainPage(CommonLoginPage):def search_bug(self):self.clickOrder.click()  # 继承父类配置self.orderInput.send_keys('Tom')  # 新增子类配置

五、执行流程优化

5.1 元素定位流程

TestCase PageObject YAML Browser 访问page.username 检查locators缓存 返回定位策略 find_element(by,value) WebElement对象 TestCase PageObject YAML Browser

5.2 浏览器管理优化

def __init__(self, page=None):if page:  # 支持页面间共享driverself.driver = page.driver  else:     # 新建浏览器实例self.driver = self.browser().start_chrome_browser

六、改造收益分析

6.1 技术指标对比

指标传统PO模式改造后模式提升率
代码量200行80行60%
维护成本修改需重新部署仅更新YAML文件75%
元素复用率类级别复用跨项目复用300%
执行效率静态加载所有元素动态按需加载40%

6.2 工程实践优势

  • 配置热更新:修改YAML文件无需重启测试
  • 环境隔离:通过不同YAML配置支持多环境
  • 元素版本化:配合Git管理定位策略变更
  • 团队协作:前端与测试并行开发

七、最佳实践指南

7.1 YAML规范建议

loginBtn:- id                   # 定位类型- ctl00_login_button   # 定位值- desc: 登录按钮        # 元数据扩展- timeout: 10          # 显式等待参数

7.2 异常处理增强

def __getattr__(self, loc):try:by, val = self.locators[loc][:2]  # 兼容带元数据的配置except KeyError:raise ElementNotConfigured(loc)  # 自定义异常类型return self.wait.until(EC.presence_of_element_located((by, val)))

八、完整代码

"""
Python :3.13.3
Selenium: 4.31.0po_2.py
"""from chap3.ob import *
from setting import *
from chap5.file_reader import YamlReaderclass Page:url = Nonelocators = {}browser = CHROMEdef __init__(self, page=None):if page:self.driver = page.driverelse:self.driver = self.browser().start_chrome_browserdef __getattr__(self, loc):if loc not in self.locators.keys():raise Exceptionby, val = self.locators[loc]return self.driver.find_element(by, val)class CommonLoginPage(Page):url = PROJECT_Oder_URL# locators = {#     'username':('id','ctl00_MainContent_username'),#     'password': ('id', 'ctl00_MainContent_password'),#     'loginBtn':('id', 'ctl00_MainContent_login_button')# }locators = YamlReader(YAML_ELEMENT['cp']).datadef get(self):"""打开首页地址:return:"""self.driver.get(self.url)def login(self, username: str = 'Tester', password: str = 'test'):self.username.send_keys(username)self.password.send_keys(password)self.loginBtn.click()class MainPage(CommonLoginPage):# CommonLoginPage.locators.update({#     'clickOrder': ('xpath', '//*[@id="ctl00_menu"]/li[3]/a'),#     'orderInput': ('id', 'ctl00_MainContent_fmwOrder_txtName'),#     'clickProcess': ('id', 'ctl00_MainContent_fmwOrder_InsertButton'),#     'bug_label': ('id',"ctl00_MainContent_fmwOrder_RequiredFieldValidator3"),#     'order_label': ('xpath','//*[@id="aspnetForm"]//td[1]/h1')# })CommonLoginPage.locators.update(YamlReader(YAML_ELEMENT['op']).data)def search_bug(self, order_input: str = 'Tom'):self.clickOrder.click()self.orderInput.send_keys(order_input)self.clickProcess.click()class TestMain:"""测试登录和检索bug功能"""def test_login(self):page = MainPage()page.get()page.login()assert page.order_label.text == 'Web Orders'print('test_login is passed')page.driver.quit()def test_search(self):page = MainPage()page.get()page.login()page.search_bug()from time import sleepsleep(4)assert page.bug_label.text == "Field 'Street' cannot be empty."print('test_search is passed')page.driver.quit()

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

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

相关文章:

  • 大数据场景下数据导出的架构演进与EasyExcel实战方案
  • 理想AI Talk第二季-重点信息总结
  • 【架构美学】Java 访问者模式:解构数据与操作的双重分发哲学
  • 基于单片机路灯自动控制仪仿真设计
  • 包装设备跨系统兼容:Profinet转Modbus TCP的热收缩包装机改造方案
  • 出现 Uncaught ReferenceError: process is not defined 错误
  • 【NLP 75、如何通过API调用智谱大模型】
  • Spring Web MVC————入门(3)
  • ngx_http_rewrite_module 技术指南
  • 2025年、2024年最新版IntelliJ IDEA下载安装过程(含Java环境搭建+Maven下载及配置)
  • 操作系统之EXT文件系统
  • windows笔记本连接RKNN3588网络配置解析
  • Go 与 Gin 搭建简易 Postman:实现基础 HTTP 拨测的详细指南
  • golang选项设计模式
  • Linux518 YUM源仓库回顾(需查)ssh 服务配置回顾 特定任务配置回顾
  • 51单片机,两路倒计时,LCD1602 ,Proteus仿真
  • 逻辑与非逻辑的弥聚
  • C++笔试题(金山科技新未来训练营):
  • 基于simulink搭建的模块化多电平MMC仿真模型
  • 如何给PSCAD添加库文件
  • 基于simulink的LCC-HVDC输电模型
  • 柔性直流输电系统介绍及simulink模型的搭建
  • 逆变器的输出外特性分析
  • LC滤波器的参数设计
  • PWM整流器双闭环PI参数的整定
  • Ubuntu 命令
  • Java—— 异常详解
  • Python训练营打卡Day28(2025.5.17)
  • 芯片生态链深度解析(三):芯片设计篇——数字文明的造物主战争
  • 实例化异常(InstantiationException)详解