当前位置:   article > 正文

【我就讲一遍】python+selenium自动化测试框架详解_python+selenium框架

python+selenium框架

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

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

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

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

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

1、测试框架分层设计

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

2、测试框架目录结构

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

3、编写用例方法

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

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

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

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

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

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

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

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

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

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

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

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

4、测试结果展示

  • HTML报告日志

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

  • 测试报告通过的日志

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

  • 邮件测试报告

行动吧,在路上总比一直观望的要好,未来的你肯定会感 谢现在拼搏的自己!如果想学习提升找不到资料,没人答疑解惑时,请及时加入扣群: 320231853,里面有各种软件测试+开发资料和技术可以一起交流学习哦。

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

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

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/空白诗007/article/detail/1001078
推荐阅读
相关标签
  

闽ICP备14008679号