【Python】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_login001
 3         title: 登录测试
 4         info: 打开抽屉首页
 5 testcase:
 6       - element_info: login-link-a
 7         find_type: ID
 8         operate_type: click
 9         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.1
 3   detail : 手机号和密码为空登录
 4   screenshot : phone_pawd_empty
 5   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 python
  2 # _*_ coding:utf-8 _*_
  3 __author__ = 'YinJia'
  4 
  5 import os,sys
  6 sys.path.append(os.path.dirname(os.path.dirname(os.path.dirname(__file__))))
  7 from config import setting
  8 from selenium.webdriver.support.select import Select
  9 from selenium.webdriver.common.action_chains import ActionChains
 10 from selenium.webdriver.common.by import By
 11 from public.page_obj.base import Page
 12 from time import sleep
 13 from public.models.GetYaml import getyaml
 14 
 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 python
 2 # _*_ coding:utf-8 _*_
 3 __author__ = 'YinJia'
 4 
 5 
 6 import os,sys
 7 sys.path.append(os.path.dirname(os.path.dirname(__file__)))
 8 import unittest,ddt,yaml
 9 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 python
 2 # _*_ coding:utf-8 _*_
 3 __author__ = 'YinJia'
 4 
 5 import os,sys
 6 sys.path.append(os.path.dirname(__file__))
 7 from config import setting
 8 import unittest,time
 9 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报告点击截图,弹出截图

  • 测试报告通过的日志

  • 自动截图存放指定的目录

  • 邮件测试报告


              【下面是我整理的2023年最全的软件测试工程师学习知识架构体系图】


一、Python编程入门到精通

二、接口自动化项目实战

三、Web自动化项目实战


四、App自动化项目实战

五、一线大厂简历


六、测试开发DevOps体系

七、常用自动化测试工具


八、JMeter性能测试

九、总结(尾部小惊喜)

生命不息,奋斗不止。每一份努力都不会被辜负,只要坚持不懈,终究会有回报。珍惜时间,追求梦想。不忘初心,砥砺前行。你的未来,由你掌握!

生命短暂,时间宝贵,我们无法预知未来会发生什么,但我们可以掌握当下。珍惜每一天,努力奋斗,让自己变得更加强大和优秀。坚定信念,执着追求,成功终将属于你!

只有不断地挑战自己,才能不断地超越自己。坚持追求梦想,勇敢前行,你就会发现奋斗的过程是如此美好而值得。相信自己,你一定可以做到!

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

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Gilisoft Video Editor——迈出剪辑的第一步

今天博主分享的是又一款剪辑软件——视频剪辑手(GiliSoft Video Editor),对剪辑视频感兴趣的小伙伴千万不要错过。这是一款专门用于视频剪辑的软件,功能比较简单,相比于专业的pr是比不了的,但是制作一些简单…

C/C++ 编程规范总结

目录 前言 一、编程规范的作用 二、规范的三种形式 三、规范的内容 1. 基本原则 原则1-1 原则1-2 原则1-3 原则1-4 原则1-5 原则1-6 原则1-7 2. 布局 规则2-1-1 规则2-1-2 规则2-1-3 规则2-1-4 规则2-1-5 规则2-1-6 规则2-2-1 规则2-2-2 规则2-2-3 建议2…

掌握iText:轻松处理PDF文档-基础篇

关于iText iText是一个强大的PDF处理库,可以用于创建、读取和操作PDF文件。它支持PDF表单、加密和签署等操作,同时支持多种字体和编码。maven的中央仓库中的最新版本是5.X,且iText5不是完全免费的,但是基础能力是免费使用的&…

pWnOS v2.0

该靶机绑定了静态IP地址 10.10.10.100,所以这里需要修改我们的网络配置!整个网段修改为10.10.10.0/24 信息收集 主机存活探测 arp-scan -l 端口信息探测 nmap -sT --min-rate 10000 -p- 10.10.10.100 (只开放了22 80端口) 服务…

2023-12-10 LeetCode每日一题(爬楼梯)

2023-12-10每日一题 一、题目编号 70. 爬楼梯二、题目链接 点击跳转到题目位置 三、题目描述 假设你正在爬楼梯。需要 n 阶你才能到达楼顶。 每次你可以爬 1 或 2 个台阶。你有多少种不同的方法可以爬到楼顶呢? 示例 1: 示例 2: 提…

【Python】手把手教你用tkinter设计图书管理登录UI界面(三)

上一篇:【Python】手把手教你用tkinter设计图书管理登录UI界面(二)-CSDN博客 下一篇: 紧接上一篇文章,继续完善项目功能:用户登录。由于老王的注册部分有亿点点复杂,还没完成,但是…

期末速成数据库极简版【分支循环函数】(4)

目录 全局变量&局部变量 局部变量定义declare 局部变量赋值select 局部变量赋值select 【1】分支结构IF 【2】分支结构CASE 简单CASE语句 搜索CASE语句 【3】循环结构While 【4】系统函数 常用字符串函数 时间函数 【5】自定义函数—标量函数 函数创建 函…

oops-framework框架 之 Excel转Json

引擎: CocosCreator 3.8.0 环境: Mac Gitee: oops-plugin-excel-to-json 注: 作者dgflash的oops-framework框架QQ群: 628575875 配置 作者dgflash在oops-framework的框架中,提供了关于Excel数据表转换为Json和TypeSc…

typora中显示除号的问题

问题 在latex中“除号( \div )” 通常用 \div。但在typora中写数学公式时,却发现 “除号” 如果使用 \div 并没有显示为 “ \div ”,而是 “ ∇ ⋅ \nabla \cdot ∇⋅ ”。 原因 typora中,\div 显示为 ∇ ⋅ \…

Html转PDF,前端JS实现Html页面导出PDF(html2canvas+jspdf)

Html转PDF,前端JS实现Html页面导出PDF(html2canvasjspdf) 文章目录 Html转PDF,前端JS实现Html页面导出PDF(html2canvasjspdf)一、背景介绍二、疑问三、所使用技术html2canvasjspdf 四、展示开始1、效果展示…

Java第21章网络通信

网络程序设计基础 网络程序设计编写的是与其他计算机进行通信的程序。Java 已经将网络程序所需要的元素封 装成不同的类,用户只要创建这些类的对象,使用相应的方法,即使不具备有关的网络支持,也可 以编写出高质量的网络…

pyinstaller 常用命令参数

PyInstaller是一个用于将Python程序打包成独立的可执行文件的工具。它可以将Python代码和所有依赖的库、资源文件等打包成一个单独的可执行文件,方便在不安装Python解释器的环境中运行。PyInstaller提供了许多参数,用于配置打包过程和生成的可执行文件的…

NSS [NSSCTF 2022 Spring Recruit]babyphp

NSS [NSSCTF 2022 Spring Recruit]babyphp 考点&#xff1a;PHP特性 开局源码直接裸奔 <?php highlight_file(__FILE__); include_once(flag.php);if(isset($_POST[a])&&!preg_match(/[0-9]/,$_POST[a])&&intval($_POST[a])){if(isset($_POST[b1])&&…

java--Date、SimpleDateFormat时间类,JDK8之前的

1.Date 代表的是日期和时间 2.SimpleDateFormat 代表简单日期格式化&#xff0c;可以用来把日期对象、时间毫秒值格式化成我们想要的形式。 3.时间格式常见符号 4.SimpleDateFormat解析字符串时间成为日期对象

Redis之IO多路复用模型

Redis之IO多路复用模型 多路复用要解决的问题 解决同步阻塞IO模型下大量线程创建导致资源的浪费问题 同步阻塞IO模式的特点就是用一个进程来处理一个网络连接(一个用户请求)&#xff0c;比如一段典型的示例代码如下。 直接调用 recv 函数从一个 socket 上读取数据。 int main…

语义分割 简介及数据集简介

参考文章 MS COCO数据集介绍以及pycocotools简单使用-CSDN博客

Linux7安装tomcat9.0.83教程

1.下载tomcat.tar.gz包 地址&#xff1a;Apache Tomcat - Apache Tomcat 9 Software Downloads 2.将包上传到linux服并解压 cd /home/local # 跳转到上传包的目录 tar -zxvf apache-tomcat-9.0.83.tar.gz # 解压包 mv apache-tomcat-9.0.83 tomcat # 重命名目录为tomcat cp …

【PUSDN】SpringBoot的jar进行解压后,替换其中的文件重新生成新的jar-SW

当你解压Spring Boot的JAR文件时&#xff0c;实际上是在打开一个压缩文件&#xff0c;类似于ZIP。你可以按照以下步骤进行替换文件并重新生成新的JAR&#xff1a; 解压原始的JAR文件&#xff1a; 使用任何ZIP工具&#xff08;如WinRAR、7-Zip或命令行工具&#xff09;&#xf…

详解异常 ! !(对异常有一个全面的认识)

【本章目标】 1. 异常概念与体系结构 2. 异常的处理方式 3. 异常的处理流程 4. 自定义异常类 1. 异常的概念与体系结构 1.1 异常的概念 在生活中&#xff0c;一个人表情痛苦&#xff0c;出于关心&#xff0c;可能会问&#xff1a;你是不是生病了&#xff0c;需要我陪你去看医…

在线测试http接口,为您解析最佳测试方法

您是否正在寻找一种方便、高效且可靠的方法来测试您的http接口&#xff1f;在这篇文章中&#xff0c;我们将为您介绍在线测试http接口的最佳方法&#xff0c;帮助您确保您的接口在各种情况下都能正常运行。 什么是http接口&#xff1f; 在开始介绍如何测试http接口之前&#x…