你一定想不到,实现一个Python+Selenium的自动化测试框架就这么简单!

爱码小士 2021-07-20 04:31:21
Python selenium


watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

首先你得知道什么是Selenium?

 

Selenium是一个基于浏览器的自动化测试工具,它提供了一种跨平台、跨浏览器的端到端的web自动化解决方案。Selenium主要包括三部分:Selenium IDE、Selenium WebDriver 和Selenium Grid。

  • Selenium IDE:Firefox的一个扩展,它可以进行录制回放,并把录制的操作以多种语言(例如java、python等)的形式导出成测试用例。

  • Selenium WebDriver:提供Web自动化所需的API,主要用作浏览器控制、页面元素选择和调试。不同的浏览器需要不同的WebDriver。

  • Selenium Grid:提供了在不同机器的不同浏览器上运行selenium测试的能力。

下面我会使用思维导图目录结构介绍基础测试框架,编写测试用例进行功能测试用例,希望对您的学习有所帮助。

 

设计思路

 

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

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

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

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

     

 

测试框架分层设计

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

 

把常见的操作和查找封装成基础类,不管是什么产品,可直接拿来复用

 

  • 业务层主要是封装对象页面类,一个页面建一个类,业务层页面继承基础层

  • 用例层针对产品页面功能进行构造模拟执行测试

  • 框架层提供基础组件,支撑整个流程执行及功能扩展,给用例层提供各页面的元素数据、用例测试数据,测试报告输出等

 

测试框架目录结构

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

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

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

 

编写用例方法

 

  •  
testinfo:- id: test_login001title: 登录测试info: 打开抽屉首页testcase:- element_info: login-link-afind_type: IDoperate_type: clickinfo: 打开登录对话框- element_info: mobilefind_type: IDoperate_type: send_keysinfo: 输入手机号- element_info: mbpwdfind_type: IDoperate_type: send_keysinfo: 输入密码- element_info: //input[@class='keeplogin']find_type: XPATHoperate_type: clickinfo: 单击取消自动登录单选框- element_info: //span[text()='登录']find_type: XPATHoperate_type: clickinfo: 单击登录按钮- element_info: userProNickfind_type: IDoperate_type: performinfo: 鼠标悬停账户菜单- element_info: //a[@class='logout']find_type: XPATHoperate_type: clickinfo: 选择退出check:- element_info: //div[@class='box-mobilelogin']/div[1]/spanfind_type: XPATHinfo: 检查输入手机号或密码,登录异常提示- element_info: userProNickfind_type: IDinfo: 成功登录- element_info: reg-link-afind_type: IDinfo: 检查退出登录是否成功login.yaml

 

例如,我们要新增登录功能测试用例:

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

 

  •  
-id: test_login001.1detail : 手机号和密码为空登录screenshot : phone_pawd_emptydata:phone: ""password: ""check :- 手机号不能为空-id: test_login001.2detail : 手机号为空登录screenshot : phone_emptydata :phone: ""password : aacheck :- 手机号不能为空-id: test_login001.3detail : 密码为空登录screenshot : pawd_emptydata :phone : 13511112222password: ""check :- 密码不能为空-id: test_login001.4detail : 非法手机号登录screenshot : phone_errordata :phone : abcpassword: aacheck :- 手机号格式不对-id: test_login001.5detail : 手机号或密码不匹配screenshot : pawd_errordata :phone : 13511112222password: aacheck :- 账号密码错误-id: test_login001.6detail : 手机号和密码正确screenshot : phone_pawd_successdata :phone : 13865439800password: ********check :- yingojalogin_data.yaml
login_data.yaml

 

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

 

  •  
#!/usr/bin/env python# _*_ coding:utf-8 _*___author__ = 'YinJia' import os,syssys.path.append(os.path.dirname(os.path.dirname(os.path.dirname(__file__))))from config import settingfrom selenium.webdriver.support.select import Selectfrom selenium.webdriver.common.action_chains import ActionChainsfrom selenium.webdriver.common.by import Byfrom public.page_obj.base import Pagefrom time import sleepfrom public.models.GetYaml import getyamltestData = getyaml(setting.TEST_Element_YAML+ '/' + 'login.yaml')class login(Page):"""用户登录页面"""url = '/'dig_login_button_loc = (By.ID, testData.get_elementinfo(0)) def dig_login(self):"""首页登录:return:"""self.find_element(*self.dig_login_button_loc).click() sleep(1)# 定位器,通过元素属性定位元素对象# 手机号输入框login_phone_loc = (By.ID,testData.get_elementinfo(1)) # 密码输入框login_password_loc = (By.ID,testData.get_elementinfo(2)) # 取消自动登录keeplogin_button_loc = (By.XPATH,testData.get_elementinfo(3)) # 单击登录login_user_loc = (By.XPATH,testData.get_elementinfo(4)) # 退出登录login_exit_loc = (By.ID, testData.get_elementinfo(5)) # 选择退出login_exit_button_loc = (By.XPATH,testData.get_elementinfo(6))def login_phone(self,phone):"""登录手机号:param username::return:"""self.find_element(*self.login_phone_loc).send_keys(phone)def login_password(self,password):"""登录密码:param password::return:"""self.find_element(*self.login_password_loc).send_keys(password) def keeplogin(self):"""取消单选自动登录:return:"""self.find_element(*self.keeplogin_button_loc).click()def login_button(self):"""登录按钮:return:"""self.find_element(*self.login_user_loc).click()def login_exit(self):"""退出系统:return:"""above = self.find_element(*self.login_exit_loc)ActionChains(self.driver).move_to_element(above).perform() sleep(2)self.find_element(*self.login_exit_button_loc).click()def user_login(self,phone,password):"""登录入口:param username: 用户名:param password: 密码:return:"""self.open()self.dig_login()self.login_phone(phone)self.login_password(password)sleep(1)self.keeplogin()sleep(1)self.login_button()sleep(1)phone_pawd_error_hint_loc = (By.XPATH,testData.get_CheckElementinfo(0))user_login_success_loc = (By.ID,testData.get_CheckElementinfo(1))exit_login_success_loc = (By.ID,testData.get_CheckElementinfo(2))# 手机号或密码错误提示def phone_pawd_error_hint(self):return self.find_element(*self.phone_pawd_error_hint_loc).text# 登录成功用户名def user_login_success_hint(self):return self.find_element(*self.user_login_success_loc).text # 退出登录def exit_login_success_hint(self):return self.find_element(*self.exit_login_success_loc).textloginPage.py

 

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

  •  
#!/usr/bin/env python# _*_ coding:utf-8 _*___author__ = 'YinJia' import os,syssys.path.append(os.path.dirname(os.path.dirname(__file__)))import unittest,ddt,yamlfrom config import settingfrom public.models import myunit,screenshotfrom public.page_obj.loginPage import loginfrom public.models.log import Logtry:f =open(setting.TEST_DATA_YAML + '/' + 'login_data.yaml',encoding='utf-8')testData = yaml.load(f)except FileNotFoundError as file:log = Log()log.error("文件不存在:{0}".format(file))@ddt.ddtclass Demo_UI(myunit.MyTest):"""抽屉新热榜登录测试"""def user_login_verify(self,phone,password):"""用户登录:param phone: 手机号:param password: 密码:return:"""login(self.driver).user_login(phone,password)def exit_login_check(self):"""退出登录:return:"""login(self.driver).login_exit()@ddt.data(*testData)def test_login(self,datayaml):"""登录测试:param datayaml: 加载login_data登录测试数据:return:"""log = Log()log.info("当前执行测试用例ID-> {0} ; 测试点-> {1}".format(datayaml['id'],datayaml['detail']))# 调用登录方法self.user_login_verify(datayaml['data']['phone'],datayaml['data']['password'])po = login(self.driver)if datayaml['screenshot'] == 'phone_pawd_success':log.info("检查点-> {0}".format(po.user_login_success_hint()))self.assertEqual(po.user_login_success_hint(), datayaml['check'][0], "成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint()))log.info("成功登录,返回实际结果是->: {0}".format(po.user_login_success_hint()))screenshot.insert_img(self.driver, datayaml['screenshot'] + '.jpg')log.info("-----> 开始执行退出流程操作")self.exit_login_check()po_exit = login(self.driver)log.info("检查点-> 找到{0}元素,表示退出成功!".format(po_exit.exit_login_success_hint()))self.assertEqual(po_exit.exit_login_success_hint(),'注册',"退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint()))log.info("退出登录,返回实际结果是->: {0}".format(po_exit.exit_login_success_hint()))else:log.info("检查点-> {0}".format(po.phone_pawd_error_hint()))self.assertEqual(po.phone_pawd_error_hint(),datayaml['check'][0] , "异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint()))log.info("异常登录,返回实际结果是->: {0}".format(po.phone_pawd_error_hint()))screenshot.insert_img(self.driver,datayaml['screenshot'] + '.jpg')if __name__=='__main__':unittest.main()login_sta.py

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

综上所述,编写用例方法只需要按以上四个步骤创建->编写即可。

执行如下主程序,可看输出的实际结果。

  •  
#!/usr/bin/env python# _*_ coding:utf-8 _*___author__ = 'YinJia' import os,syssys.path.append(os.path.dirname(__file__))from config import settingimport unittest,timefrom package.HTMLTestRunner import HTMLTestRunnerfrom public.models.newReport import new_reportfrom public.models.sendmail import send_mail# 测试报告存放文件夹,如不存在,则自动创建一个report目录 if not os.path.exists(setting.TEST_REPORT):os.makedirs(setting.TEST_REPORT + '/' + "screenshot")def add_case(test_path=setting.TEST_DIR):"""加载所有的测试用例"""discover = unittest.defaultTestLoader.discover(test_path, pattern='*_sta.py')return discoverdef run_case(all_case,result_path=setting.TEST_REPORT):"""执行所有的测试用例"""now = time.strftime("%Y-%m-%d %H_%M_%S")filename = result_path + '/' + now + 'result.html'fp = open(filename,'wb')runner = HTMLTestRunner(stream=fp,title='抽屉新热榜UI自动化测试报告',description='环境:windows 7 浏览器:chrome',tester='Jason')runner.run(all_case)fp.close()report = new_report(setting.TEST_REPORT)#调用模块生成最新的报告send_mail(report) #调用发送邮件模块if __name__ =="__main__":cases = add_case()run_case(cases)

 

测试结果展示

 

HTML报告日志

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

 

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

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

测试报告通过的日志

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

自动截图存放指定的目录

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

邮件测试报告

 

watermark,size_16,text_QDUxQ1RP5Y2a5a6i,color_FFFFFF,t_100,g_se,x_10,y_10,shadow_90,type_ZmFuZ3poZW5naGVpdGk=

 


 

 



版权声明
本文为[爱码小士]所创,转载请带上原文链接,感谢
https://blog.51cto.com/u_14126611/2895129

  1. 【推荐】常见的Python数据可视化库
  2. 基于Python3.3+Selenium3.0框架实战Web自动化测试实战(百度网站实测项目)
  3. 【django轻量级框架】用Mysql的各种项目响应速度慢?一招解决!
  4. 【django轻量级框架】Django项目导入css,js,images等静态文件
  5. 【django轻量级框架】HTML上传文件拦截到本地
  6. 【深度学习入门到精通系列】留一交叉验证法Python实现(看不懂你来打我~!)
  7. Python机器学习笔记:sklearn库的学习
  8. python【Opencv计算机视觉库】opencv模块cv2常用函数用法(全)
  9. python基础练习(七)
  10. python基础练习(八)
  11. python基础练习(九)
  12. python基础练习(十)
  13. python新增练习(一)
  14. python新增练习(二)
  15. python新增练习(三)
  16. Python中的排序sorted(d.items(), key=lambda x: x[1])
  17. python 判断字母大小写
  18. python绘制科赫雪花(递归)
  19. 解决cmd命令查看python版本“python不是内部命令或外部命令,也不是可执行程序解决方案”的问题
  20. Python Numpy介绍
  21. python —pandas库常用函数
  22. Python应用matplotlib绘图简介
  23. Python matplotlib高级绘图详解
  24. 入门训练 Fibonacci数列-python实现
  25. Python -二维数组定义
  26. python二进制相加
  27. Python文本处理:解析json格式的数据
  28. 查看python安装路径
  29. Python编程之计算生态
  30. Python-turtle标准库知识小结(python绘图工具)
  31. Python-time标准库知识小结
  32. Python-random标准库知识小结
  33. python安装第三方库的三种方法
  34. python程序的控制结构
  35. Python程序的函数和代码复用
  36. python之组合数据类型
  37. python【力扣LeetCode算法题库】300 最长上升子序列(动态规划)
  38. python【力扣LeetCode算法题库】695- 岛屿的最大面积(深搜)
  39. python【力扣LeetCode算法题库】面试题 01.06-字符串压缩
  40. python【力扣LeetCode算法题库】1160-拼写单词
  41. python【力扣LeetCode算法题库】836- 矩形重叠
  42. python【力扣LeetCode算法题库】409-最长回文串(数学 计数器)
  43. python【力扣LeetCode算法题库】面试题40- 最小的k个数
  44. python【力扣LeetCode算法题库】945- 使数组唯一的最小增量
  45. python【力扣LeetCode算法题库】365- 水壶问题(裴蜀等式)
  46. python【力扣LeetCode算法题库】876- 链表的中间结点
  47. python【力扣LeetCode算法题库】面试题 17.16- 按摩师(DP)
  48. python【力扣LeetCode算法题库】892-三维形体的表面积
  49. python【力扣LeetCode算法题库】999-车的可用捕获量(DFS)
  50. python【力扣LeetCode算法题库】914. 卡牌分组(reduce & collections.Counter)
  51. python【力扣LeetCode算法题库】820- 单词的压缩编码
  52. python【力扣LeetCode算法题库】1162- 地图分析(BFS)
  53. python【力扣LeetCode算法题库】面试题62- 圆圈中最后剩下的数字(约瑟夫环)
  54. python【力扣LeetCode算法题库】912- 排序数组
  55. python【力扣LeetCode算法题库】1111- 有效括号的嵌套深度
  56. python【力扣LeetCode算法题库】289- 生命游戏
  57. python【力扣LeetCode算法题库】12- 整数转罗马数字(打表 模拟)
  58. python【数据结构与算法】内置函数enumerate(枚举) 函数(看不懂你来打我)
  59. python【力扣LeetCode算法题库】13- 罗马数字转整数
  60. python【数据结构与算法】内置函数 zip() 函数(看不懂你来打我)