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

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

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

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

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

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

1、测试框架分层设计

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

2、测试框架目录结构

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

3、编写用例方法


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)

4、测试结果展示

  • HTML报告日志

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

  • 测试报告通过的日志

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

  • 邮件测试报告

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

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

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

  • 16
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值