一、PO设计模式+数据驱动+Unittest
本篇通过改造完善《WebUI之PO设计模式与Unittest》https://www.jianshu.com/p/14f759f8613b的内容进行了一点调整,多出了读取excel(xls格式文件)与数据驱动
二、PO模式超简单实例
(一)项目结构
BaseMethod:存放一些定义的通用方法,如读写excel,csv。本篇未用
Data:存放准备的测试数据
Page:BasePage放基本的元素定位与操作方法,SearchPage
TestCase:存放测试用例
TestResult:测试结果存放目录,一般放截图,测试结果写入数据文件(CSV或excel),本篇未用。
(二)BaseMethod.Method
这是前面其他篇章的内容,直接复制过来,放入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根据IDE自动出来的名称是_int,这个不是初始化的构造方法,我踩过的坑,你们就不要踩了。
class Page(object):
"""
基类,PO模式的页面都继承这个类
"""
def __init__(self,driver,base_url):
#driver,base_url是在实例化的时候需要传入的参数
#注意__init__这个名称不要写错或者写漏,如果这里写漏了,什么都看不到
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类再次封装页面元素与操作方法,相当于将基本定位与操作方法与元素,输入文本等进行了隔离
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
注意:代码中的方法,每次执行测试用例都会打开一个浏览器
这篇分享的文章中有一种只打开一次浏览器的方法https://www.jianshu.com/p/3ccb8236c23a,但是这种方法存在缺陷,就是一个py文件中只能有一个测试集和一个测试用例,用来做注册测试用例很合适;做一个页面中有多个多个输入框的用例就需要改造了。原因是def setUpClass(cls) -> None:没有self,定义了一个全局变量driver。后面进行改造,不使用全局变量。
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数据表.xls', 0, 0)
@data(*input_data)
#若不使用*解包,那么input_data里面的内容会被当成一个数据全部输入文本框
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)
改造:在输入框内输入数据,不重复打开浏览器,节约测试时间
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数据表.xls', 0, 0)
@data(*input_data)
#若不使用*解包,那么input_data里面的内容会被当成一个数据全部输入文本框
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框架重复运行多次之后,跑过去只执行单个测试用例,而不执行测试集或者py文件,回报错误,但只要右击左侧导航栏需要执行的py文件直接执行就会有问题了