WebUI之PO设计模式+数据驱动+Unittest(简易版)

一、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),本篇未用。


image.png

(二)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文件直接执行就会有问题了

你可能感兴趣的:(WebUI之PO设计模式+数据驱动+Unittest(简易版))