Python+Selenium 之数据驱动测试

一、数据驱动

模式的测试好处相比普通模式的测试就显而易见了吧!使用数据驱动的模式,可以根据业务分解测试数据,只需定义变量,使用外部或者自定义的数据使其参数化,从而避免了使用之前测试脚本中固定的数据。可以将测试脚本与测试数据分离,使得测试脚本在不同数据集合下高度复用。不仅可以增加复杂条件场景的测试覆盖,还可以极大减少测试脚本的编写与维护工作。

下面将使用Python下的数据驱动模式(ddt)库,结合unittest库以数据驱动模式创建百度搜索的测试。

ddt库包含一组类和方法用于实现数据驱动测试。可以将测试中的变量进行参数化。

可以通过python自带的pip命令进行下载并安装:pip install ddt .

二、一个简单的数据驱动测试

为了创建数据驱动测试,需要在测试类上使用@ddt装饰符,在测试方法上使用@data装饰符。@data装饰符把参数当作测试数据,参数可以是单个值、列表、元组、字典。对于列表,需要用@unpack装饰符把元组和列表解析成多个参数。

下面实现百度搜索测试,传入搜索关键词和期望结果,代码如下:

import unittest

from selenium import webdriver

from ddt import ddt, data, unpack

@ddt

class SearchDDT(unittest.TestCase):

    '''docstring for SearchDDT'''

    def setUp(self):

        self.driver = webdriver.Chrome()

        self.driver.implicitly_wait(30)

        self.driver.maximize_window()

        self.driver.get("https://www.baidu.com")

    # specify test data using @data decorator

    @data(('python', 'PyPI'))

    @unpack

    def test_search(self, search_value, expected_result):

        search_text = self.driver.find_element_by_id('kw')

        search_text.clear()

        search_text.send_keys(search_value)

        search_button = self.driver.find_element_by_id('su')

        search_button.click()

        tag = self.driver.find_element_by_link_text("PyPI").text

        self.assertEqual(expected_result, tag)

    def tearDown(self):

        self.driver.quit()

if __name__ == '__main__':

    unittest.main(verbosity=2)

在test_search()方法中,search_value与expected_result两个参数用来接收元组解析的数据。当运行脚本时,ddt把测试数据转换为有效的python标识符,生成名称为更有意义的测试方法。结果如下:

1、使用外部数据的数据驱动测试

如果外部已经存在了需要的测试数据,如一个文本文件、电子表格或者数据库,那也可以用ddt来直接获取数据并传入测试方法进行测试。

下面将借助外部的CSV(逗号分隔值)文件和EXCLE表格数据来实现ddt。

2、通过CSV获取数据

同上在@data装饰符使用解析外部的CSV(testdata.csv)来作为测试数据(代替之前的测试数据)。其中数据如下

接下来,先要创建一个get_data()方法,其中包括路径(这里默认使用当前路径)、CSV文件名。调用CSV库去读取文件并返回一行数据。再使用@ddt及@data实现外部数据驱动测试百度搜索,代码如下:

import csv, unittest

from selenium import webdriver

from ddt import ddt, data, unpack

def get_data(file_name):

    # create an empty list to store rows

    rows = []

    # open the CSV file

    data_file = open(file_name, "r")

    # create a CSV Reader from CSV file

    reader = csv.reader(data_file)

    # skip the headers

    next(reader, None)

    # add rows from reader to list

    for row in reader:

        rows.append(row)

    return rows

@ddt

class SearchCSVDDT(unittest.TestCase):

    def setUp(self):

        self.driver = webdriver.Chrome()

        self.driver.implicitly_wait(30)

        self.driver.maximize_window()

        self.driver.get("https://www.baidu.com")

    # get test data from specified csv file by using the get_data funcion

    @data(*get_data('testdata.csv'))

    @unpack

    def test_search(self, search_value, expected_result):

        search_text = self.driver.find_element_by_id('kw')

        search_text.clear()

        search_text.send_keys(search_value)

        search_button = self.driver.find_element_by_id('su')

        search_button.click()

        tag = self.driver.find_element_by_link_text("PyPI").text

        self.assertEqual(expected_result, tag)

    def tearDown(self):

        self.driver.quit()

if __name__ == '__main__':

    unittest.main(verbosity=2)

测试执行时,@data将调用get_data()方法读取外部数据文件,并将数据逐行返回给@data。执行的结果也同上

3、通过Excel获取数据

测试中经常用Excle存放测试数据,同上在也可以使用@data装饰符来解析外部的CSV(testdata.csv)来作为测试数据(代替之前的测试数据)。其中数据如下:

接下来,先要创建一个get_data()方法,其中包括路径(这里默认使用当前路径)、EXCEL文件名。调用xlrd库去读取文件并返回数据。再使用@ddt及@data实现外部数据驱动测试百度搜索,代码如下:

import xlrd, unittest

from selenium import webdriver

from ddt import ddt, data, unpack

def get_data(file_name):

    # create an empty list to store rows

    rows = []

    # open the CSV file

    book = xlrd.open_workbook(file_name)

    # get the frist sheet

    sheet = book.sheet_by_index(0)

    # iterate through the sheet and get data from rows in list

    for row_idx in range(1, sheet.nrows):  #iterate 1 to maxrows

        rows.append(list(sheet.row_values(row_idx, 0, sheet.ncols)))

    return rows

@ddt

class SearchEXCLEDDT(unittest.TestCase):

    def setUp(self):

        self.driver = webdriver.Chrome()

        self.driver.implicitly_wait(30)

        self.driver.maximize_window()

        self.driver.get("https://www.baidu.com")

    # get test data from specified excle spreadsheet by using the get_data funcion

    @data(*get_data('TestData.xlsx'))

    @unpack

    def test_search(self, search_value, expected_result):

        search_text = self.driver.find_element_by_id('kw')

        search_text.clear()

        search_text.send_keys(search_value)

        search_button = self.driver.find_element_by_id('su')

        search_button.click()

        tag = self.driver.find_element_by_link_text("PyPI").text

        self.assertEqual(expected_result, tag)

    def tearDown(self):

        self.driver.quit()

if __name__ == '__main__':

    unittest.main(verbosity=2)

与上面读取CVS文件一样,测试执行时,@data将调用get_data()方法读取外部数据文件,并将数据逐行返回给@data。执行的结果也同上~

如果想从数据库的库表中获取数据,同样也需要一个get_data()方法,并且通过DB相关的库来连接数据库、SQL查询来获取测试数据。以上内容希望对你有帮助,有被帮助到的朋友欢迎点赞,评论。如果对软件测试、接口测试、自动化测试、面试经验交流。感兴趣可以加软件测试交流:829792258,还会有同行一起技术交流。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,684评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,143评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,214评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,788评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,796评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,665评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,027评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,679评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,346评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,664评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,766评论 1 331
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,412评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,015评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,974评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,203评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,073评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,501评论 2 343