当前位置: 首页 > 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()

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

相关文章:

  • Elasticsearch基础篇-java程序通过RestClient操作es
  • BUUCTF——ReadlezPHP
  • Android 手写签名功能详解:从原理到实践
  • 第五部分:阶段项目 4:构建 RESTful API 服务器
  • 关于机器学习的实际案例
  • 【Java ee初阶】jvm(2)
  • 股票数据源对接技术指南:印度尼西亚、印度、韩国
  • ubuntu22.04搭建ROS2环境
  • C++_数据结构_哈希表(hash)实现
  • 阶段四 项目1-苍穹外卖 第一章 Git
  • 论文学习:《引入TEC - LncMir,通过对RNA序列的深度学习来预测lncRNA - miRNA的相互作用》
  • Java面试深度解析:微服务与云原生技术应用场景详解
  • 菜鸟之路Day32一一多表查询,事物,索引
  • Ubuntu 20.04 报错记录: Matplotlib 无法使用 OpenCV 的 libqxcb.so
  • 76.有符号数累加运算
  • 从坏道扫描到错误修复:HD Tune实战指南
  • 小白到高手的人工智能学习笔记之初步了解pytorch
  • Ubuntu24.04下安装ISPConfig全过程记录
  • AM32电调学习解读八:无感驱动相位波形解析
  • 架构思维:构建高并发扣减服务_分布式无主架构
  • 经济日报:政府采购监管篱笆要扎得更牢
  • 私家车跑“顺风”出事故,意外险赔不赔?
  • 蒲慕明院士:好的科普应以“质疑、讨论公众关切的科学问题”为切入点
  • 俄外长与美国务卿通电话,讨论俄美接触等问题
  • 新闻1+1丨强对流天气频繁组团来袭,该如何更好应对?
  • 梅花奖在上海|湘剧《夫人如见》竞梅,长沙文旅来沪推广