自动化测试:PO模式介绍及案例

PO(Page Object)设计模式是一种面向对象( 页面对象)的设计模式,将测试对象及单个的测试步骤封装在每个Page对象以page为单位进行管理。

优点

  1. 可以使代码复用
  2. 降低维护成本
  3. 提高程序可读性和编写效率。
  4. 可以将页面定位和业务操作分开,测试对象(元素对象)和测试脚本(用例脚本)
  5. 提高用例的可维护
非PO模式PO模式
面向过程的线性脚本POM把页面元素定位和业务操作流程分开,实现松耦合
复用性性差UI元素的改变不需要修改业务逻辑代码,只需要找到对应的PO页修改定位即可,数据代码分离
维护性差PO能使代码更具有可读性,高复用性、可维护性
PO三层模式:

主要分三层:

1.base层(对象库层):page页面一些公共的方法。如:初始化、元素定位、点击、输入、获取文本、截图等方法;

2.page层(操作层):封装对元素的操作。将每个涉及的元素操作单独封装一个操作方法,然后根据需求组装操作步骤,如登录方法=输入帐号+输入密码+点击登录三个操作进行组装;

3.scripts层(业务层):导包调用 page页面,使用单元测试框架对业务逻辑进行封装测试。如:实现登录,直接调用page组装的登陆方法即可。

三者的关系:page层继承base层,scripts层调用page层

案例:

项目结构介绍:

创建项目,如下图

1. 构建基础的 BasePage 对象层

创建driver,浏览器驱动封装

 
  1. # encoding='UTF-8'

  2. # 浏览器启动

  3. from selenium import webdriver

  4. def browser():

  5. driver=webdriver.Chrome()

  6. # driver.get("http://www.baidu.com")

  7. return driver

创建myuni.py文件,初始化封装,

定义一个继承自unittest.TestCase的测试用例类

定义setUp和tearDown,这两个方法与junit相同,即如果定义了则会在每个测试case执行前先执行setUp方法,执行完毕后执行tearDown方法。

 
  1. # encoding='UTF-8'

  2. import unittest

  3. from driver import *

  4. class StartEnd(unittest.TestCase):

  5. def setUp(self):

  6. self.driver=browser()

  7. self.driver.implicitly_wait(10)

  8. self.driver.maximize_window()

  9. def tearDown(self):

  10. self.driver.quit()

创建function.py文件,截图功能

 
  1. # encoding='UTF-8'

  2. import os

  3. from selenium import webdriver

  4. # 截图

  5. def insert_img(driver,filename):

  6. func_path=os.path.dirname(__file__)

  7. # print(func_path)

  8. base_dir=os.path.dirname(func_path)

  9. # print(base_dir)

  10. # 将路径转化为字符串

  11. base_dir=str(base_dir)

  12. # 对路径的字符串进行替换

  13. base_dir=base_dir.replace("\\","/")

  14. # print(base_dir)

  15. # 获取项目文件的要目录路径

  16. base=base_dir.split('/Website')[0]

  17. # print(base)

  18. # 指定截图存放路径(注意路径最后要加/)

  19. filepath=base+'/Website/test_report/screnshot/'+filename

  20. driver.get_screenshot_as_file(filepath)

  21. if __name__=='__main__':

  22. driver=webdriver.Chrome()

  23. driver.get("http://www.sogou.com")

2. 构建首页的 Page 层(操作层)

创建BasePage.py文件,判断打开的页面是否是预期的页面

 
  1. # encoding='UTF-8'

  2. from time import sleep

  3. class Page():

  4. def __init__(self,driver):

  5. self.driver=driver

  6. self.base_url="http://localhost"

  7. self.timeout=10

  8. def _open(self,url):

  9. url_=self.base_url+url

  10. # print("这个页面url是:%s"%url_)

  11. self.driver.maximize_window()

  12. self.driver.get(url_)

  13. # sleep(2)

  14. assert self.driver.current_url==url_,'这不是我们想要的地址'

  15. def open(self):

  16. self._open(self.url)

  17. def find_element(self,*loc):

  18. return self.driver.find_element(*loc)

创建LoginPage.py文件,页面元素定位封装

 
  1. # encoding='UTF-8'

  2. from BasePage import *

  3. from selenium.webdriver.common.by import By

  4. class LoginPage(Page):

  5. url='/news/'

  6. username_loc=(By.NAME,'username')

  7. password_loc=(By.NAME,'password')

  8. submit_loc=(By.NAME,'Submit')

  9. def type_username(self,username):

  10. self.find_element(*self.username_loc).send_keys(username)

  11. def type_password(self,password):

  12. self.find_element(*self.password_loc).send_keys(password)

  13. def type_submit(self):

  14. self.find_element(*self.submit_loc).click()

  15. def login_action(self,username,password):

  16. self.open()

  17. self.type_username(username)

  18. self.type_password(password)

  19. self.type_submit()

  20. loginPass_loc=(By.LINK_TEXT,'我的空间')

  21. loginErr_loc=(By.LINK_TEXT,'加入收藏')

  22. def type_loginPass_hint(self):

  23. return self.find_element(*self.loginPass_loc).text

  24. def type_loginErr_hit(self):

  25. return self.find_element(*self.loginErr_loc).text

3.构建业务层
  • 创建test_login.py文件
  • 导包:function、myunit、LoginPage
  • 创建LoginTest类,继承myunit.StartEnd,初始化方法
  • 定义测试用例,名字以test开头,unittest会自动将test开头的方法放入测试用例集中
  • 实现登录,调用LoginPage组装的登陆方法,输入用户名、密码、点击登录、断言,截图
 
  1. # encoding='UTF-8'

  2. import unittest

  3. from model import function,myunit

  4. from page_object.LoginPage import *

  5. from time import sleep

  6. class LoginTest(myunit.StartEnd):

  7. def test_login1_normal(self):

  8. print("test_login1_normal测试开始")

  9. po=LoginPage(self.driver)

  10. po.login_action("yuruyi","12345678")

  11. sleep(5)

  12. self.assertEqual(po.type_loginPass_hint(),'我的空间')

  13. function.insert_img(self.driver,"login_normal.png")

  14. print("test_login1_normal执行结束")

  15. def test_login2_PasswdError(self):

  16. print("test_login2_PasswdError测试开始")

  17. po=LoginPage(self.driver)

  18. po.login_action("yuruyi","1234567")

  19. sleep(5)

  20. self.assertEqual(po.type_loginErr_hit(),'加入收藏')

  21. function.insert_img(self.driver,"login_Err.png")

  22. print("test_login2_PasswdError执行结束")

  23. def test_login3_empty(self):

  24. print("test_login3_empty测试开始")

  25. po = LoginPage(self.driver)

  26. po.login_action("", "")

  27. sleep(5)

  28. self.assertEqual(po.type_loginErr_hit(), '加入收藏')

  29. function.insert_img(self.driver, "login_empty.png")

  30. print("test_login3_empty执行结束")

  31. if __name__=='__main__':

  32. unittest.main()

常用断言方法:

4. 构建用例集,执行文件,输出自动化测试报告

在测试用例、测试文件比较多的时候,使用统一的主测试执行文件进行测试用例的执行非常方便,这就需要结合discover方法和TextTestRunner进行。

 
  1. # encoding='UTF-8'

  2. import unittest

  3. # 测试报告模板

  4. from HTMLTestRunnerCN import HTMLTestRunner

  5. import time

  6. report_dir='./test_report'

  7. test_dir='./test_case'

  8. print("start run test case")

  9. discover=unittest.defaultTestLoader.discover(test_dir,pattern="test_login.py")

  10. now=time.strftime("%Y-%m-%d %H_%M_%S")

  11. report_name=report_dir+'/'+now+'result.html'

  12. print("start write report..")

  13. #使用runner运行器运行测试集

  14. with open(report_name,'wb')as f:

  15. runner=HTMLTestRunner(stream=f,title="Test Report",description="localhost login test")

  16. runner.run(discover)

  17. f.close()

  18. print("Test end")

测试报告模板

 

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

这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!有需要的小伙伴可以点击下方小卡片领取   

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值