Selenium自动化测试框架详解

🍅 点击文末小卡片 ,免费获取软件测试全套资料,资料在手,涨薪更快 

设计思路

本文整理归纳以往的工作中用到的东西,现汇总成基础测试框架提供分享。

框架采用python3 + selenium3 + PO + yaml + ddt + unittest等技术编写成基础测试框架,能适应日常测试工作需要。

1、使用Page Object模式将页面定位和业务操作分开,分离测试对象(元素对象)和测试脚本(用例脚本),一个页面建一个对象类,提高用例的可维护性;

2、使用yaml管理页面控件元素数据和测试用例数据。例如元素ID等发生变化时,不需要去修改测试代码,只需要在对应的页面元素yaml文件中修改即可;

3、分模块管理,互不影响,随时组装,即拿即用。

GitHub项目地址:GitHub - yingoja/DemoUI: selenium UI自动化测试框架

测试框架分层设计

  • 把常见的操作和查找封装成基础类,不管是什么产品,可直接拿来复用
  • 业务层主要是封装对象页面类,一个页面建一个类,业务层页面继承基础层
  • 用例层针对产品页面功能进行构造摸拟执行测试
  • 框架层提供基础组件,支撑整个流程执行及功能扩展,给用例层提供各页面的元素数据、用例测试数据,测试报告输出等

测试框架目录结构


如下思维导图目录结构介绍:

编写用例方法

 1 testinfo:2       - id: test_login0013         title: 登录测试4         info: 打开抽屉首页5 testcase:6       - element_info: login-link-a7         find_type: ID8         operate_type: click9         info: 打开登录对话框
10       - element_info: mobile
11         find_type: ID
12         operate_type: send_keys
13         info: 输入手机号
14       - element_info: mbpwd
15         find_type: ID
16         operate_type: send_keys
17         info: 输入密码
18       - element_info: //input[@class='keeplogin']
19         find_type: XPATH
20         operate_type: click
21         info: 单击取消自动登录单选框
22       - element_info: //span[text()='登录']
23         find_type: XPATH
24         operate_type: click
25         info: 单击登录按钮
26       - element_info: userProNick
27         find_type: ID
28         operate_type: perform
29         info: 鼠标悬停账户菜单
30       - element_info: //a[@class='logout']
31         find_type: XPATH
32         operate_type: click
33         info: 选择退出
34 check:
35       - element_info: //div[@class='box-mobilelogin']/div[1]/span
36         find_type: XPATH
37         info: 检查输入手机号或密码,登录异常提示
38       - element_info: userProNick
39         find_type: ID
40         info: 成功登录
41       - element_info: reg-link-a
42         find_type: ID
43         info: 检查退出登录是否成功

例如,我们要新增登录功能测试用例:首先,只需在testyaml目录下新增一个页面对象yaml文件,参考login.yaml格式编写即可。这些文件是提供给封装页面对象类调用并执行定位识别操作。

 1 -2   id: test_login001.13   detail : 手机号和密码为空登录4   screenshot : phone_pawd_empty5   data:6     phone: ""7     password: ""8   check :9      - 手机号不能为空
10 -
11   id: test_login001.2
12   detail : 手机号为空登录
13   screenshot : phone_empty
14   data :
15     phone: ""
16     password : aa
17   check :
18     - 手机号不能为空
19 -
20   id: test_login001.3
21   detail : 密码为空登录
22   screenshot : pawd_empty
23   data :
24     phone : 13511112222
25     password: ""
26   check :
27     - 密码不能为空
28 -
29   id: test_login001.4
30   detail : 非法手机号登录
31   screenshot : phone_error
32   data :
33     phone : abc
34     password: aa
35   check :
36     - 手机号格式不对
37 -
38   id: test_login001.5
39   detail : 手机号或密码不匹配
40   screenshot : pawd_error
41   data :
42     phone : 13511112222
43     password: aa
44   check :
45     - 账号密码错误
46 -
47   id: test_login001.6
48   detail : 手机号和密码正确
49   screenshot : phone_pawd_success
50   data :
51     phone : 13865439800
52     password: ********
53   check :
54     - yingoja
55 
56 login_data.yaml

其次,在testdata目录下新增一个login_data.yaml文件提供给登录接口传参的测试数据,编写格式参考login_data.yaml文件。

 1 #!/usr/bin/env python2 # _*_ coding:utf-8 _*_3 __author__ = 'YinJia'4 5 import os,sys6 sys.path.append(os.path.dirname(os.path.dirname(os.path.dirname(__file__))))7 from config import setting8 from selenium.webdriver.support.select import Select9 from selenium.webdriver.common.action_chains import ActionChains10 from selenium.webdriver.common.by import By11 from public.page_obj.base import Page12 from time import sleep13 from public.models.GetYaml import getyaml14 15 testData = getyaml(setting.TEST_Element_YAML + '/' + 'login.yaml')16 17 class login(Page):18     """19     用户登录页面20     """21     url = '/'22     dig_login_button_loc = (By.ID, testData.get_elementinfo(0))23     def dig_login(self):24         """25         首页登录26         :return:27         """28         self.find_element(*self.dig_login_button_loc).click()29         sleep(1)30 31     # 定位器,通过元素属性定位元素对象32     # 手机号输入框33     login_phone_loc = (By.ID,testData.get_elementinfo(1))34     # 密码输入框35     login_password_loc = (By.ID,testData.get_elementinfo(2))36     # 取消自动登录37     keeplogin_button_loc = (By.XPATH,testData.get_elementinfo(3))38     # 单击登录39     login_user_loc = (By.XPATH,testData.get_elementinfo(4))40     # 退出登录41     login_exit_loc = (By.ID, testData.get_elementinfo(5))42     # 选择退出43     login_exit_button_loc = (By.XPATH,testData.get_elementinfo(6))44 45     def login_phone(self,phone):46         """47         登录手机号48         :param username:49         :return:50         """51         self.find_element(*self.login_phone_loc).send_keys(phone)52 53     def login_password(self,password):54         """55         登录密码56         :param password:57         :return:58         """59         self.find_element(*self.login_password_loc).send_keys(password)60 61     def keeplogin(self):62         """63         取消单选自动登录64         :return:65         """66         self.find_element(*self.keeplogin_button_loc).click()67 68     def login_button(self):69         """70         登录按钮71         :return:72         """73         self.find_element(*self.login_user_loc).click()74 75     def login_exit(self):76         """77         退出系统78         :return:79         """80         above = self.find_element(*self.login_exit_loc)81         ActionChains(self.driver).move_to_element(above).perform()82         sleep(2)83         self.find_element(*self.login_exit_button_loc).click()84 85     def user_login(self,phone,password):86         """87         登录入口88         :param username: 用户名89         :param password: 密码90         :return:91         """92         self.open()93         self.dig_login()94         self.login_phone(phone)95         self.login_password(password)96         sleep(1)97         self.keeplogin()98         sleep(1)99         self.login_button()
100         sleep(1)
101 
102     phone_pawd_error_hint_loc = (By.XPATH,testData.get_CheckElementinfo(0))
103     user_login_success_loc = (By.ID,testData.get_CheckElementinfo(1))
104     exit_login_success_loc = (By.ID,testData.get_CheckElementinfo(2))
105 
106     # 手机号或密码错误提示
107     def phone_pawd_error_hint(self):
108         return self.find_element(*self.phone_pawd_error_hint_loc).text
109 
110     # 登录成功用户名
111     def user_login_success_hint(self):
112         return self.find_element(*self.user_login_success_loc).text
113 
114     # 退出登录
115     def exit_login_success_hint(self):
116         return self.find_element(*self.exit_login_success_loc).text

然后,在page_obj目录下新增一个loginPage.py文件,是用来封装登录页面对象类,执行登录测试流程操作。

 1 #!/usr/bin/env python2 # _*_ coding:utf-8 _*_3 __author__ = 'YinJia'4 5 6 import os,sys7 sys.path.append(os.path.dirname(os.path.dirname(__file__)))8 import unittest,ddt,yaml9 from config import setting
10 from public.models import myunit,screenshot
11 from public.page_obj.loginPage import login
12 from public.models.log import Log
13 
14 try:
15     f =open(setting.TEST_DATA_YAML + '/' + 'login_data.yaml',encoding='utf-8')
16     testData = yaml.load(f)
17 except FileNotFoundError as file:
18     log = Log()
19     log.error("文件不存在:{0}".format(file))
20 
21 @ddt.ddt
22 class Demo_UI(myunit.MyTest):
23     """抽屉新热榜登录测试"""
24     def user_login_verify(self,phone,password):
25         """
26         用户登录
27         :param phone: 手机号
28         :param password: 密码
29         :return:
30         """
31         login(self.driver).user_login(phone,password)
32 
33     def exit_login_check(self):
34         """
35         退出登录
36         :return:
37         """
38         login(self.driver).login_exit()
39 
40     @ddt.data(*testData)
41     def test_login(self,datayaml):
42         """
43         登录测试
44         :param datayaml: 加载login_data登录测试数据
45         :return:
46         """
47         log = Log()
48         log.info("当前执行测试用例ID-> {0} ; 测试点-> {1}".format(datayaml['id'],datayaml['detail']))
49         # 调用登录方法
50         self.user_login_verify(datayaml['data']['phone'],datayaml['data']['password'])
51         po = login(self.driver)
52         if datayaml['screenshot'] == 'phone_pawd_success':
53             log.info("检查点-> {0}".format(po.user_login_success_hint()))
54             self.assertEqual(po.user_login_success_hint(), datayaml['check'][0], "成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint()))
55             log.info("成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint()))
56             screenshot.insert_img(self.driver, datayaml['screenshot'] + '.jpg')
57             log.info("-----> 开始执行退出流程操作")
58             self.exit_login_check()
59             po_exit = login(self.driver)
60             log.info("检查点-> 找到{0}元素,表示退出成功!".format(po_exit.exit_login_success_hint()))
61             self.assertEqual(po_exit.exit_login_success_hint(), '注册',"退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint()))
62             log.info("退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint()))
63         else:
64             log.info("检查点-> {0}".format(po.phone_pawd_error_hint()))
65             self.assertEqual(po.phone_pawd_error_hint(),datayaml['check'][0] , "异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint()))
66             log.info("异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint()))
67             screenshot.insert_img(self.driver,datayaml['screenshot'] + '.jpg')
68 
69 if __name__=='__main__':
70     unittest.main()

最后,在testcase目录下创建测试用例文件login_sta.py,采用ddt数据驱动读取yaml测试数据文件

综上所述,编写用例方法只需要按以上四个步骤创建->编写即可。执行如下主程序,可看输出的实际结果。

 1 #!/usr/bin/env python2 # _*_ coding:utf-8 _*_3 __author__ = 'YinJia'4 5 import os,sys6 sys.path.append(os.path.dirname(__file__))7 from config import setting8 import unittest,time9 from package.HTMLTestRunner import HTMLTestRunner
10 from public.models.newReport import new_report
11 from public.models.sendmail import send_mail
12 
13 # 测试报告存放文件夹,如不存在,则自动创建一个report目录
14 if not os.path.exists(setting.TEST_REPORT):os.makedirs(setting.TEST_REPORT + '/' + "screenshot")
15 
16 def add_case(test_path=setting.TEST_DIR):
17     """加载所有的测试用例"""
18     discover = unittest.defaultTestLoader.discover(test_path, pattern='*_sta.py')
19     return discover
20 
21 def run_case(all_case,result_path=setting.TEST_REPORT):
22     """执行所有的测试用例"""
23     now = time.strftime("%Y-%m-%d %H_%M_%S")
24     filename =  result_path + '/' + now + 'result.html'
25     fp = open(filename,'wb')
26     runner = HTMLTestRunner(stream=fp,title='抽屉新热榜UI自动化测试报告',
27                             description='环境:windows 7 浏览器:chrome',
28                             tester='Jason')
29     runner.run(all_case)
30     fp.close()
31     report = new_report(setting.TEST_REPORT) #调用模块生成最新的报告
32     send_mail(report) #调用发送邮件模块
33 
34 if __name__ =="__main__":
35     cases = add_case()
36     run_case(cases)

测试结果展示

HTML报告日志

HTML报告点击截图,弹出截图

测试报告通过的日志

自动截图存放指定的目录

邮件测试报告

最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

这些资料,对于做【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴我走过了最艰难的路程,希望也能帮助到你!凡事要趁早,特别是技术行业,一定要提升技术功底。

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

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

相关文章

ChangeCLIP环境配置

看到有个现成的dockerfile,先试试 ok首先需要root权限的用户 才能用docker,其次要外网,要不然有些东西好像下载不了 (失败) 那就直接配吧 我看12服务器上有个openmmlab的环境,先基于这个环境吧 # 用lx账…

【MATLAB源码-第272期】基于matlab的OMP算法的毫米波MIMO通信系统的混合波束成形仿真。

操作环境: MATLAB 2022a 1、算法描述 在现代无线通信系统中,随着频谱资源的日益紧张,毫米波(mmWave)技术成为5G及未来通信系统的重要组成部分。毫米波频段的宽频带提供了远超传统微波频段的频谱资源,能够…

【python】OpenCV—findContours(4.3)

文章目录 1、功能描述2、代码实现3、完整代码4、结果展示5、涉及到的库函数5.1、cv2.Canny5.2 cv2.boxPoints 6、参考 1、功能描述 找出图片中的轮廓,拟合轮廓外接椭圆和外接矩阵 2、代码实现 导入必要的库,固定好随机种子 import cv2 as cv import …

直播推流和拉流--系统篇

今天实现一下直播推流和拉流。服务器端使用opencloudos8系统。顺便把我之前写的小系统弄上去跑跑,搭建个git服务器,使用ssh协议,密钥方式。 先展示一下在iphone上推流效果图 再展示下在谷歌浏览器上的拉流效果图,safari浏览器和微…

安全芯片 OPTIGA TRUST M 使用介绍与示例(基于STM32裸机)

文章目录 目的资料索引硬件电路软件框架介绍数据存储框架移植框架使用 使用示例示例地址与硬件连接通讯测试功能测试 总结 目的 OPTIGA TRUST M 是英飞凌推出的安全芯片,芯片通提供了很多 slot ,用于存放各类安全证书、密钥、用户数据等,内置…

数据结构 之 二叉树遍历 ------中序(根)遍历 和 后序(根)遍历(六)

提示:本篇章主要讲解数据结构中树的相关知识。 文章目录 中序(根)遍历二叉树(LTR)后序(根)遍历二叉树(LRT)中根遍历二叉树的递归算法 (重要)后序遍历二叉树的…

数据结构 ——— 二叉树的概念及结构

目录 二叉树的概念 特殊的二叉树 一、满二叉树 二、完全二叉树 二叉树的概念 二叉树树示意图: 从以上二叉树示意图可以看出: 二叉树每个节点的度不大于 2 ,那么整个二叉树的度也不大于 2 ,但是也不是每个节点都必须有 2 个…

IDEA解决 properties 文件乱码问题

博主介绍: 计算机科班人,全栈工程师,掌握C、C#、Java、Python、Android等主流编程语言,同时也熟练掌握mysql、oracle、sqlserver等主流数据库,具有丰富的项目经验和开发技能。提供相关的学习资料、程序开发、技术解答、…

MySQL 9从入门到性能优化-创建触发器

【图书推荐】《MySQL 9从入门到性能优化(视频教学版)》-CSDN博客 《MySQL 9从入门到性能优化(视频教学版)(数据库技术丛书)》(王英英)【摘要 书评 试读】- 京东图书 (jd.com) MySQL9数据库技术_夏天又到了…

电子邮件营销平台教程:从零开始营销指南!

电子邮件营销平台功能详解?哪个电子邮件营销平台好? 无论是初创企业还是成熟品牌,掌握电子邮件营销平台的技巧,都能有效提升品牌影响力和销售业绩。MailBing将为你提供一份详尽的电子邮件营销平台教程,帮助你从零开始…

软件行业似乎要消失了

关注卢松松,会经常给你分享一些我的经验和观点。 软件行业是企业倒闭和裁员降薪的重灾区,其实比网站、站长、电商群体更惨。 据赛迪发布,354家软件上市公司财报数据显示:上半年合计利润6.9亿,同比下跌91.62%&#x…

windows 10 clion配置32位c++环境

主要就是这一个地方,需要32位的mingw32,mingw32可以在红框的蓝色download下载

计算机视觉-对极几何

1 基本概念 对极几何(Epipolar Geometry)描述的是两幅视图之间的内在射影关系,与外部场景无关,只依赖于摄像机内参数和这两幅视图之间的相对位姿 两视图的对极几何可以理解为图像平面与以基线为轴的平面束相交的几何关系&#xf…

leetcode344. Reverse String

Write a function that reverses a string. The input string is given as an array of characters s. You must do this by modifying the input array in-place with O(1) extra memory. Example 1: Input: s [“h”,“e”,“l”,“l”,“o”] Output: [“o”,“l”,“l”…

ssm016基于 Java Web 的校园驿站管理系统(论文+源码)_kaic

毕 业 设 计(论 文) 题目:校园驿站管理系统的设计与实现 摘 要 互联网发展至今,无论是其理论还是技术都已经成熟,而且它广泛参与在社会中的方方面面。它让信息都可以通过网络传播,搭配信息管理工具可以很好…

鸿蒙自定义加载 LoadingDialog

代码如下: Component export struct LoadingDialog {Prop title: stringbuild() {Stack() {Column() {LoadingProgress().color(Color.White).width(100).height(100)Text(this.title).fontSize(18).fontColor(0xffffff).margin({ top: 8 }).visibility(this.title …

【AI开源项目】Dify- 轻松打造可持续运营的 GPT 系列的 AI应用 —— 全面解析LLMOps平台

文章目录 什么是Dify?Dify的名称由来 了解LLMOpsDify的核心功能兼容多种LLMs Dify的优势完全开源核心能力 如何安装Dify快速启动使用Helm Chart在Kubernetes上部署自定义配置 使用Dify创建AI应用第一步:创建应用程序第二步:编写和调试提示词第…

【HTML】——VS Code 基本使用入门和常见操作,新手小白也能看懂

阿华代码,不是逆风,就是我疯 你们的点赞收藏是我前进最大的动力!! 希望本文内容能够帮助到你!! 目录 零:HTML开发工具VSCode的使用 1:创建项目 2:创建格式模板&#x…

基于springboot+vue实现的公考知识学习平台 (源码+L文+ppt)4-103

4.1 系统功能结构设计 根据对公考知识学习平台的具体需求分析,把系统可以划分为几个不同的功能模块:管理员可以对系统首页、个人中心、用户管理、讲师管理、在线咨询管理、学习资料管理、讲座信息管理、讲座预约管理、学习论坛、练习自测管理、试题管理…

计算结构力学:多自由度振动系统

本文以笔记的形式记录计算结构力学的若干基础知识。 注1:限于研究水平,分析难免不当,欢迎批评指正。 注2:文章内容会不定期更新。 预修1:线性代数 1. 标准特征值 复矩阵Schur分解:对于复矩阵&#xff0c…