一、PO設(shè)計(jì)模式+數(shù)據(jù)驅(qū)動+Unittest
????本篇通過改造完善《WebUI之PO設(shè)計(jì)模式與Unittest》http://www.reibang.com/p/14f759f8613b的內(nèi)容進(jìn)行了一點(diǎn)調(diào)整,多出了讀取excel(xls格式文件)與數(shù)據(jù)驅(qū)動
二、PO模式超簡單實(shí)例
(一)項(xiàng)目結(jié)構(gòu)
BaseMethod:存放一些定義的通用方法怔匣,如讀寫excel熟掂,csv。本篇未用
Data:存放準(zhǔn)備的測試數(shù)據(jù)
Page:BasePage放基本的元素定位與操作方法港令,SearchPage
TestCase:存放測試用例
TestResult:測試結(jié)果存放目錄淡喜,一般放截圖秕磷,測試結(jié)果寫入數(shù)據(jù)文件(CSV或excel),本篇未用炼团。
(二)BaseMethod.Method
這是前面其他篇章的內(nèi)容澎嚣,直接復(fù)制過來疏尿,放入Method.py,直接用就行币叹,如果想要通過Sheet名稱讀取润歉,將sheet_by_index改為sheet_by_name即可
from xlutils.copy import copy
import xlrd
def write_excel_xls(filename,sheet_index,row_num,col_num,content):
book=xlrd.open_workbook(filename)
book_copy=copy(book)
sheet=book_copy.get_sheet(sheet_index)
sheet.write(row_num,col_num,content)
book_copy.save(filename)
def read_excel_xls(filename,sheet_index,col_num):
"""
:param filename:
:param sheet_index: Sheet索引模狭,從0開始
:param col_num: 列索引颈抚,從0開始
:return:
"""
text=xlrd.open_workbook(filename).sheet_by_index(sheet_index).col_values(col_num,start_rowx=1)
return text
(三)BasePage
????基本頁面類,定義所有的元素的定位與操作方法嚼鹉,瀏覽器操作贩汉,按鍵操作等,獲取cookie等锚赤,不用一次寫到位匹舞,用到什么了,就往里面加线脚,慢慢就成為自己的庫了赐稽。
注意:
????_init_方法不要寫錯,是雙下橫線浑侥,init不要寫錯姊舵,_in根據(jù)IDE自動出來的名稱是_int,這個(gè)不是初始化的構(gòu)造方法寓落,我踩過的坑括丁,你們就不要踩了。
class Page(object):
"""
基類伶选,PO模式的頁面都繼承這個(gè)類
"""
def __init__(self,driver,base_url):
#driver,base_url是在實(shí)例化的時(shí)候需要傳入的參數(shù)
#注意__init__這個(gè)名稱不要寫錯或者寫漏史飞,如果這里寫漏了,什么都看不到
self.driver=driver
self.base_url=base_url
self.timeout=30
def GetPage(self):
self.driver.get(self.base_url)
def find_element(self,*locator):
return self.driver.find_element(locator)
def input_text(self,locator_method,element,value):
self.driver.find_element(locator_method,element).send_keys(value)
def click(self,locator_method,element):
return self.driver.find_element(locator_method,element).click()
def getTitle(self):
print(self.driver.title)
return self.driver.title
def driver_quit(self):
self.driver.quit()
(四)SearchPage
????SearchPage通過繼承BasePage類再次封裝頁面元素與操作方法仰税,相當(dāng)于將基本定位與操作方法與元素构资,輸入文本等進(jìn)行了隔離
from Page.BasePage import Page
from selenium.webdriver.common.by import By
class SearchPage(Page):
def searchBox_input(self,text):
return self.input_text(locator_method=By.ID,element='kw',value=text)
def search_btn_click(self):
self.click(locator_method=By.ID,element='su')
(五)TestCase.Case
注意:代碼中的方法,每次執(zhí)行測試用例都會打開一個(gè)瀏覽器
????這篇分享的文章中有一種只打開一次瀏覽器的方法http://www.reibang.com/p/3ccb8236c23a陨簇,但是這種方法存在缺陷蚯窥,就是一個(gè)py文件中只能有一個(gè)測試集和一個(gè)測試用例,用來做注冊測試用例很合適塞帐;做一個(gè)頁面中有多個(gè)多個(gè)輸入框的用例就需要改造了拦赠。原因是def setUpClass(cls) -> None:沒有self,定義了一個(gè)全局變量driver葵姥。后面進(jìn)行改造荷鼠,不使用全局變量。
from selenium import webdriver
from Page.SearchPage import SearchPage
import unittest,time
from ddt import ddt,data,unpack
from BaseMethod import Method
@ddt
class Test_search_page(unittest.TestCase):
def setUp(self) -> None:
self.driver=webdriver.Chrome()
self.base_url = 'https://www.baidu.com/'
self.search_page=SearchPage(self.driver,self.base_url)
input_data = Method.read_excel_xls('../Data/CVS數(shù)據(jù)表.xls', 0, 0)
@data(*input_data)
#若不使用*解包榔幸,那么input_data里面的內(nèi)容會被當(dāng)成一個(gè)數(shù)據(jù)全部輸入文本框
def test_searchPage(self,text):
self.search_page.GetPage()
self.search_page.searchBox_input(text)
self.search_page.search_btn_click()
self.search_page.getTitle()
def tearDown(self) -> None:
time.sleep(5)
self.driver.quit()
if __name__ == '__main__':
unittest.main(verbosity=2)
改造:在輸入框內(nèi)輸入數(shù)據(jù)允乐,不重復(fù)打開瀏覽器矮嫉,節(jié)約測試時(shí)間
from selenium import webdriver
from Page.SearchPage import SearchPage
import unittest,time
from ddt import ddt,data,unpack
from BaseMethod import Method
@ddt
class Test_search_page(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
cls.driver = webdriver.Chrome()
base_url = 'https://www.baidu.com/'
cls.search_page = SearchPage(cls.driver, base_url)
# def setUp(self) -> None:
# self.driver = webdriver.Chrome()
# self.base_url = 'https://www.baidu.com/'
# self.search_page=SearchPage(self.driver,self.base_url)
input_data = Method.read_excel_xls('../Data/CVS數(shù)據(jù)表.xls', 0, 0)
@data(*input_data)
#若不使用*解包,那么input_data里面的內(nèi)容會被當(dāng)成一個(gè)數(shù)據(jù)全部輸入文本框
def test_searchPage(self,text):
# base_url = 'https://www.baidu.com/'
# search_page = SearchPage(self.driver, base_url)
self.search_page.GetPage()
self.search_page.searchBox_input(text)
self.search_page.search_btn_click()
self.search_page.getTitle()
self.driver.back()
@classmethod
def tearDownClass(cls) -> None:
time.sleep(5)
cls.driver.quit()
if __name__ == '__main__':
unittest.main(verbosity=2)
(六)注意
????使用Unittest框架重復(fù)運(yùn)行多次之后牍疏,跑過去只執(zhí)行單個(gè)測試用例蠢笋,而不執(zhí)行測試集或者py文件,回報(bào)錯誤鳞陨,但只要右擊左側(cè)導(dǎo)航欄需要執(zhí)行的py文件直接執(zhí)行就會有問題了