
1、PO模式的思想
- 原理:
- 将页面的元素定位和元素行为封装成一个page类
- 类的属性:元素的定位
- 类的行为:元素的操作
- 页面对象和测试用例分离
- 测试用例:
- 调用所需要页面对象中的行为,组成测试用例
- 测试用例中,只需要含有页面函数的调用和断言,不应该出现元素定位等其他的操作,如果写测试用例中出现需要其他的额外的操作,可以想办法封装到页面对象或者元素定位中
- 好处:
- 当某个页面的元素发生变化,只需要修改该页面对象中的代码即可,测试用例不需要修改
- 提高代码的重用率,结构清晰,维护代码容易
- 测试用例发生变化是,不需要或者只需要修改少数页面对象代码即可
2、使用unittest框架实现PO模式
- 新建一个包,命名为PageObjects,包中用来封装各个页面的功能
- 页面对象封装的类中,函数用来实现页面的功能,在类的初始化函数中,使用传参(driver)的方式完成,页面只需要实现对应功能即可,具体传入什么样的测试,在测试用例的前置条件中实现,可以提高函数的重用率
-
页面对象封装函数
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.by import By
class LoginPage:
def __init__(self,driver):
self.driver=driver def login(self,username,pwd,remember_user=True):
# 等待定位元素出现
WebDriverWait(self.driver,10).until(EC.visibility_of_element_located((By.XPATH,"//input[@name='phone']")))
# 输入用户名
self.driver.find_element_by_xpath("//input[@name='phone']").send_keys(username)
# 输入密码
self.driver.find_element_by_xpath("//input[@name='password']").send_keys(pwd)
# 定位记住手机号元素
remember_ele=self.driver.find_element_by_xpath("//input[@name='remember_me']")
# 判断时候记住手机号
if remember_user==True:
remember_ele.checked=True
else:
remember_ele.checked=False
# 点击登录
self.driver.find_element_by_xpath("//button[text()='登录']").click() -
测试用例调用页面对象函数,代码如下
1 def test_login_success(self):
2 # 2、步骤
3 # 实例化LoginPage类
4 lg=LoginPage(self.driver)
5 # 调用login方法
6 lg.login("登录账号","登录密码")
7 # 3、断言
8 self.assertTrue(IndexPage(self.driver).isExist_logout_ele()) - Index_page页面封装函数代码
from selenium.webdriver.support import expected_conditions as EC
from selenium.webdriver.support.wait import WebDriverWait
from selenium.webdriver.common.by import By
class IndexPage:
def __init__(self,driver):
self.driver=driver def isExist_logout_ele(self):
try:
WebDriverWait(self.driver,10).until(EC.visibility_of_element_located((By.XPATH,"//a[text()='退出']")))
return True
except:
return False - 测试用例的前置条件代码
import unittest
from selenium import webdriver
from PageObjects.login_page import LoginPage
from PageObjects.index_page import IndexPage
class TestLogin(unittest.TestCase): def setUp(self):
# 1、前置条件
url="http://ip:host/Index/login.html"
self.driver=webdriver.Chrome()
self.driver.maximize_window()
self.driver.get(url) - 运行测试用例可以在测试用例最后面使用一下代码,然后运行该代码
if __name__ == '__main__':
unittest.main() - 运行结果如下,表示运行成功
-
- 页面对象封装的类中,函数用来实现页面的功能,在类的初始化函数中,使用传参(driver)的方式完成,页面只需要实现对应功能即可,具体传入什么样的测试,在测试用例的前置条件中实现,可以提高函数的重用率
- 新建一个包,命名为TestCases,包中用来写测试用例,测试用例中用到那个页面对象中的函数时,只需要调用页面对象中的函数就可以了
- 新建一个包,命名为TestDatas,用来写测试数据,将测试数据的类导入到测试用例模块中,进行使用,这样的话,测试数据便于管理,测试数据添加、修改、删除不用修改测试用例
- 在TestCases下面新建一个公用的数据管理.py文件Common_Datas.py和模块数据的.py文件login_datas.py,如下t图:
#Common_Datas.py
web_login_url="http://ip:port/Index/login.html" #login_datas.py
#正常场景---测试数据
success_data={"user":"","pwd":""} # 异常用例--手机号格式不正确(大于11位,小于11位,为空,不在号码段)
error_data=[
{"user": "", "pwd": "","check":"请输入正确的手机号"},
{"user": "", "pwd": "","check":"请输入正确的手机号"},
{"user": "", "pwd": "","check":"请输入正确的手机号"},
{"user": "", "pwd": "","check":"请输入手机号"},
{"user": "", "pwd": "","check":"请输入密码"},
{"user": "", "pwd": "","check":"请输入手机号"},
] - 在测试用例模块使用数据管理模块中的数据
import unittest
from selenium import webdriver
from PageObjects.login_page import LoginPage
from PageObjects.index_page import IndexPage
from TestDatas import Common_Datas as CD
from TestDatas import login_datas as ld
from ddt import ddt,data @ddt
class TestLogin(unittest.TestCase):
@classmethod
def setUpClass(cls):
# 1、前置条件
cls.driver=webdriver.Chrome()
cls.driver.maximize_window()
cls.driver.get(CD.web_login_url)
# 实例化LoginPage类
cls.lg=LoginPage(cls.driver)
@classmethod
def tearDownClass(cls):
cls.driver.quit() def tearDown(self):
self.driver.refresh()
pass
# 正常用例--登录成功
def test_login_1_success(self):
# 2、步骤 # 调用login方法
self.lg.login(ld.success_data['user'],ld.success_data['pwd'])
# 3、断言
self.assertTrue(IndexPage(self.driver).isExist_logout_ele())
# 异常用例--手机号码格式不正确(大于11位,小于11位,为空,不在号码段) ddt
@data(*ld.error_data)
def test_login_0_user_wrongFormat(self,item):
self.lg.login(item['user'],item['pwd'])
self.assertEqual(self.lg.get_errorMsg_form_loginArea(),item['check'])备注:
setUpClass(cls)函数和tearDownClass(cls)只在整个用例开始和结束时运行一次,必须使用@classmethod来修饰,如上面的异常测试用例有6条,在用例开始前和结束时只运行一次
setUp(self)函数和tearDown(self)在每个测试用例开始和结束时运行,如果使用这两个函数,上面的6条用例,每运行一个,就会运行一次这两个函数,总共运行6次
tearDown(self)可以和setUpClass(cls)函数和tearDownClass(cls)函数结合使用,如上面的界面刷新self.driver.refresh()和页面关闭cls.driver.quit()
- 测试用例类型相同的测试数据可以设计在一起,使用ddt在测试用例运行时分别执行测试用例
- 使用unittest框架,运行测试用例,如下:
import unittest
from TestCases.test_login import TestLogin
from HTMLTestRunnerCN import HTMLTestReportCN
# 创建一个容器,用来存测试用例
suite=unittest.TestSuite()
# 加载测试用例的类的实例
loader=unittest.TestLoader()
# 将测试用例加载到suite容器中
suite.addTest(loader.loadTestsFromTestCase(TestLogin))
# 打开文件,用来写测试报告
with open("test_result.html",'wb') as file: # runner=unittest.TextTestRunner(verbosity=1) runner=HTMLTestReportCN( stream=file, verbosity=2, title="web自动化测试", description="第一个web自动化测试", tester="wsk")
# 运行测试用例
runner.run(suite) - 运行结果,如下图:
在写测试用例时,如果使用unittest框架可以使用函数名来控制代码的执行顺序,如:
ok test_login_0_user_wrongFormat_6 (TestCases.test_login.TestLogin) -------------------- 0
ok test_login_1_success (TestCases.test_login.TestLogin)------------------------------------------1 前面都一样,0比1小,先运行
- 在TestCases下面新建一个公用的数据管理.py文件Common_Datas.py和模块数据的.py文件login_datas.py,如下t图: