unittest框架

1.通过unittest框架创建测试

1.必须继承于unittest.TestCase类
2.可以定义setUp和tearDown方法进行初始化,每条测试用例开始或结束会执行。也可以使用setUpClass和tearDownClass来进行初始化,整个程序开始或结束会执行。
3.所有测试方法必须以test开头。测试方法会在运行时自动被调用。
4.可以pycharm自带的unittest框架运行,Run ‘Unittest for…’。也可以以普通方式运行。
在这里插入图片描述

  • setUp:每个测试方法执行前执行一次。
  • tearDown:每个测试方法结束后执行一次。
  • setUpClass:整个测试执行前执行一次。
  • tearDownClass: 整个测试执行后执行一次。

2.通过unittest框架添加断言

断言是为了判断实际结果与预期结果是否一致,任何测试都是需要断言的,否则就是操作,不能算自动化测试。在执行的过程中,断言失败了,就不会往下执行。
常见断言

1.assertIn

assertIn(key, container, message)
# key:在给定容器中检查其存在性的字符串
# container:在其中搜索关键字符串的字符串
# message:作为测试消息失败时显示的消息的字符串语句。可选项。

如果 key 包含在container字符串中,它将返回true,否则返回false,并提示message。

2.assertNotIn

3.assertEqual

assertEqual(firstValue, secondValue, message)
# firstValue按函数比较中使用的任何类型的变量
# secondValue:按函数比较时使用的任何类型的变量
# message:作为测试消息失败时显示的消息的字符串语句。
assertEqual(1, 1)

如果两个输入值相等,则assertEqual()将返回true,否则返回false。

4. assertNotEqual

5.assertTrue

assertTrue(testValue, message)
# testValue:布尔类型的变量,用于按功能比较
# message:作为测试消息失败时显示的消息的字符串语句。

如果测试值是true,则assertTrue()将返回true,否则返回false。

6.assertFalse

总结:assertIn和assertEqual比较常用

3.自动化用例管理TestLoader类的用法

测试用例有几百个,有时候只执行一部分,比如执行冒烟测试用例,回归测试用例,每次迭代的测试范围可能不一样。
TestLoader类常见方法

1.loadTestsFromTestCase()

smoketests.py 冒烟测试用例

import unittest
class SmokeTests(unittest.TestCase):  # 测试类必须继承unittest.TestCase类
      # 测试方法命名必须以test开头
        def testSk1(self):  # 测试用例1
            self.assertIn('hello', 'hell world')

        def testSk2(self):  # 测试用例2
            self.assertNotIn('hello', 'world')

requirementtests.py 需求测试用例

import unittest
class RequirementTests(unittest.TestCase):  # 测试类必须继承unittest.TestCase类
      # 测试方法命名必须以test开头
        def testRmt1(self):  # 测试用例1
            self.assertEqual(1, 1)

        def testRmt2(self):  # 测试用例2
            self.assertEqual(1, 1)

regressiontests.py 回归测试用例

import unittest
class RegressionTests(unittest.TestCase):  # 测试类必须继承unittest.TestCase类
      # 测试方法命名必须以test开头
        def testRg(self):  # 测试用例1
            self.assertIn(1, 1)

        def testRg(self):  # 测试用例2
            self.assertNotIn(1, 1)

performtests.py 执行用例

from smoketests import SmokeTests
from requirementtests import RequirementTests
from regressiontests import RegressionTests
import unittest
if __name__ == '__main__':
    sk = unittest.defaultTestLoader.loadTestsFromTestCase(SmokeTests) # 执行冒烟测试类里面的测试用例
    rq = unittest.defaultTestLoader.loadTestsFromTestCase(RequirementTests) # 执行需求测试类里面的测试用例
    rg = unittest.defaultTestLoader.loadTestsFromTestCase(RegressionTests) # 执行回归测试类里面的测试用例
    suite = unittest.TestSuite([sk])  # 执行一个测试类
    suite = unittest.TestSuite([sk, rq, rg])  # 执行冒烟,需求,回归测试
    unittest.TextTestRunner.run(suite)

2.loadTestsFromModule()

从一个python文件里面导入测试方法
performtests.py 执行用例

import smoketests, requirementtests, regressiontests 
import unittest
if __name__ == '__main__':
    names = [smoketests, requirementtests, regressiontests]  # 存放需要执行的测试类
    modules = []
    for item in names:
    	module = unittest.defaultTestLoader.loadTestsFromModule(item)
    	modules.append(module)
    suite = unittest.TestSuite(modules)  
    unittest.TextTestRunner.run(suite)

3.loadTestsFromNames()

1)执行测试类里面的部分测试用例,意思是执行冒烟,需求,回归测试用例里面的几个测试用例
performtests.py 执行用例

from smoketests import SmokeTests
from requirementtests import RequirementTests
from regressiontests import RegressionTests
import unittest
if __name__ == '__main__':
	names = ['smoketests.SmokeTests.testSk1', 'requirementtests.RequirementTests.testRmt1', 'regressiontests.RegressionTests.testRg1']  # 注意这里必须要从文件名.类名.方法写完整路径
	perform_test = unittest.defaultTestLoader.loadTestsFromNames(names)
	suite = unittest.TestSuite(perform_test) 
    unittest.TextTestRunner.run(suite)

2)如果想执行某个功能模块,这个模块在冒烟测试用例中有一些用例。
smoketests.py 冒烟测试用例

import unittest
class SmokeTests(unittest.TestCase):  # 测试类必须继承unittest.TestCase类
      # 测试方法命名必须以test开头,同一个模块测试方法命名有相同的字段
      # 登录模块的测试用例
        def testLoginSk1(self):  # 测试用例1
            self.assertIn('hello', 'hell world1')
		def testLoginSk2(self):  # 测试用例2
            self.assertIn('hello', 'hell world2')
        def testLoginSk3(self):  # 测试用例3
            self.assertIn('hello', 'hell world3')
      # 注册模块的测试用例
        def testregistSk1(self):  # 测试用例4
            self.assertNotIn('hello', 'world')
        def testregistSk2(self):  # 测试用例5
            self.assertNotIn('hello', 'world')

performtests.py 执行用例

from smoketests import SmokeTests
import unittest
def getFullTestCaseNames(names):
	full_nams = []
	for item in names:
		if 'Login' in item:
			full_names.append('smoketests.SmokeTests.' + item)
if __name__ == '__main__':
	names = unittest.defaultTestLoader.getTestCaseNames(SmokeTests)
	perform_test = unittest.defaultTestLoader.loadTestsFromNames(getFullTestCaseNames(names))
	suite = unittest.TestSuite(perform_test) 
    unittest.TextTestRunner.run(suite)

4.discover() 最常用

项目中最常用的方法,根据测试路径去选择执行哪些测试用例,冒烟,需求,回归测试用例用包分类存放用例
cases包 存放测试文件smoke包 ,requirement包,regression包
smoke包 存放冒烟测试用例的py文件
requirement包 存放冒烟测试用例的py文件
regression包 存放冒烟测试用例的py文件

performtests.py 执行用例

if __name__ == '__main__':
	case_dir = 'cases'  # cases包
	perform_test = unittest.defaultTestLoader.discover(case_dir + '/smoke', pattern='*.py')  # cases包下的smoke包下的所有py文件
	suite = unittest.TestSuite(perform_test) 
    unittest.TextTestRunner.run(suite)

4.unittest智能封装等待

webdriver有强制等待,显示等待,隐式等待,详解见web自动化测试-等待

import unittest
from selenium import webdriver
from selenium.common.exceptions import NoSuchFrameException
from selenium.webdriver.support.wait import WebDriverWait

class waitTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.driver.get('')

    def tearDown(self) -> None:
        self.driver.quit()

    def find_element(self, locator):
    # 封装driver.find_element方法
    # locator参数是元素定位
        try:
            element = WebDriverWait(self.driver, 30 ,0.5).until(lambda x: x.find_element(*locator))
            return element
        except NoSuchFrameException as e:
            print('Error details:{}'.format(e.args[0])) # e.args[0]指错误的原因
            
    def test1(self):  # 测试用例1,这里元素操作就可以不用等待
        self.find_element(('id', 'username')).send_keys('admin')
        self.find_element(('id', 'password')).send_keys('123456')
        self.find_element(('id', 'login')).click()
        self.find_element(('partial link text', '会议'))

5.unittest数据驱动

利用不同的测试数据来测试相同的场景,为了提高代码的重用性,增加代码效率而采用一种代码编写的方法,叫参数化,也就是数据驱动。
比如登录操作,需要用正确账号密码登录,不正确账号密码登录,测试方法中的等价类方法,需要测试有效等价类,无效等价类。
在unittest框架中使用参数化,需要安装ddt包。
参数化的几种形式

1.参数值为单个参数形式

@data(1, 2, 3, 4)

from selenium import webdriver
import unittest
from ddt import ddt, data, unpack


@ddt  # 必须要在类前加上这个装饰器
class LoginTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.url = ''

    def tearDown(self) -> None:
        self.driver.quit()
    
    # 数据为单个值
    @data(1, 2, 3, 4)
    def test_data(self, value):
        print(value)

结果输出:1 2 3 4

2.参数值为组合参数形式

@data(
(‘’, ‘123456’, ‘9988’, ‘用户名或密码错误’),
(‘admin’, ‘’, ‘9988’, ‘用户名或密码错误’),
(‘system’, ‘123456’, ‘9988’, ‘用户名或密码错误’),
(‘admin’, ‘7573890’, ‘9988’, ‘用户名或密码错误’),
(‘admin’, ‘123456’, ‘gre5’, ‘验证码错误’)
)
@unpack

logintest.py 登录功能

from selenium import webdriver
import unittest
from ddt import ddt, data, unpack


@ddt  # 必须要在类前加上这个装饰器
class LoginTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.url = ''

    def tearDown(self) -> None:
        self.driver.quit()

    # 登录成功的测试用例,类名以test开头
    def testLoginSuccess(self):
        self.driver.get(self.url)
        self.driver.find_element('id', 'username').send_keys('admin')  # 输入账号
        self.driver.find_element('id', 'username').send_keys('123456')  # 输入密码
        self.driver.find_element('id', 'verification').send_keys('9988')  # 输入验证码
        self.driver.find_element('xpath', '//input[@value="Login"]').click()  # 点击登录
        loginname = self.driver.find_element('id', 'loginName').text  # 登录进去显示账号名
        self.assertEqual('admin', loginname)  # 判断是否登录成功

    # 登录失败的测试数据组合
    @data(
        ('', '123456', '9988', '用户名或密码错误'),
        ('admin', '', '9988', '用户名或密码错误'),
        ('system', '123456', '9988', '用户名或密码错误'),
        ('admin', '7573890', '9988', '用户名或密码错误'),
        ('admin', '123456', 'gre5', '验证码错误')
    )
    @unpack  # 解包,将data里面的5种组合拆开成username, password, verification, errmsg参数传进下面函数中
    # 登录失败的测试用例,执行5次
    def testLoginFailed(self, username, password, verification, err):
        self.driver.get(self.url)
        self.driver.find_element('id', 'username').clear()
        self.driver.find_element('id', 'username').send_keys(username)  # 输入账号
        self.driver.find_element('id', 'username').clear()
        self.driver.find_element('id', 'username').send_keys(password)  # 输入密码
        self.driver.find_element('id', 'verification').clear()
        self.driver.find_element('id', 'verification').send_keys(verification)  # 输入验证码
        self.driver.find_element('xpath', '//input[@value="Login"]').click()  # 点击登录
        errmsg = self.driver.find_element('xpath', '//div[contains(text(), "{}")').format(err).text  # 登录失败显示的提示框
        self.assertEqual(errmsg, err)  # 判断是登录失败提示信息是否正确


if __name__ == '__main__':
    test = unittest.defaultTestLoader.loadTestsFromTestCase(LoginTest)
    suite = unittest.TestSuite(test)
    unittest.TextTestRunner().run(suite)

缺点:如果数据比较多,写在代码里面,维护困难。

3.从函数中返回参数值

将测试数据单独放到一个函数中
testdata.py

class TestData:
    
    @classmethod
    def get_data(cls):
        testdata = [
            ('', '123456', '9988', '用户名或密码错误'),
            ('admin', '', '9988', '用户名或密码错误'),
            ('system', '123456', '9988', '用户名或密码错误'),
            ('admin', '7573890', '9988', '用户名或密码错误'),
            ('admin', '123456', 'gre5', '验证码错误')
        ]
        return testdata

logintest.py

from selenium import webdriver
import unittest
from ddt import ddt, data, unpack
from testdata import TestData

@ddt  # 必须要在类前加上这个装饰器
class LoginTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.url = ''

    def tearDown(self) -> None:
        self.driver.quit()

    # 登录成功的测试用例,类名以test开头
    def testLoginSuccess(self):
        self.driver.get(self.url)
        self.driver.find_element('id', 'username').send_keys('admin')  # 输入账号
        self.driver.find_element('id', 'username').send_keys('123456')  # 输入密码
        self.driver.find_element('id', 'verification').send_keys('9988')  # 输入验证码
        self.driver.find_element('xpath', '//input[@value="Login"]').click()  # 点击登录
        loginname = self.driver.find_element('id', 'loginName').text  # 登录进去显示账号名
        self.assertEqual('admin', loginname)  # 判断是否登录成功

    # 登录失败的测试数据组合
    @data(*TestData.get_data())  # 这里的*号是解包的意思,将testdata列表解包出5个元组,然后穿给@data,如果不加*,就是将testdata列表值整个传给@data
    @unpack  # 解包,将data里面的5种组合拆开成username, password, verification, errmsg参数传进下面函数中
    # 登录失败的测试用例,执行5次
    def testLoginFailed(self, username, password, verification, err):
        self.driver.get(self.url)
        self.driver.find_element('id', 'username').clear()
        self.driver.find_element('id', 'username').send_keys(username)  # 输入账号
        self.driver.find_element('id', 'username').clear()
        self.driver.find_element('id', 'username').send_keys(password)  # 输入密码
        self.driver.find_element('id', 'verification').clear()
        self.driver.find_element('id', 'verification').send_keys(verification)  # 输入验证码
        self.driver.find_element('xpath', '//input[@value="Login"]').click()  # 点击登录
        errmsg = self.driver.find_element('xpath', '//div[contains(text(), "{}")').format(err).text  # 登录失败显示的提示框
        self.assertEqual(errmsg, err)  # 判断是登录失败提示信息是否正确

4. 从文件中返回参数值(最优,最常用)

建一个文件存放测试数据

data.txt

,123456,9988,用户名或密码错误
admin,,9988,用户名或密码错误
system,123456,9988,用户名或密码错误
admin,7573890,9988,用户名或密码错误
admin,123456,gre5,验证码错误

testdata.py 读取测试数据文件里面的数据的方法

class TestData:

    @classmethod
    def get_data_from_file(cls, path):
        rows = []
        with open(path, 'r', encoding='utf-8') as f:
            for line in f:
                user_data = line.strip().split(',')
                rows.append(user_data)
        return rows

logintest.py

from selenium import webdriver
import unittest
from ddt import ddt, data, unpack
from testdata import TestData

@ddt  # 必须要在类前加上这个装饰器
class LoginTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.url = ''

    def tearDown(self) -> None:
        self.driver.quit()

    # 登录成功的测试用例,类名以test开头
    def testLoginSuccess(self):
        self.driver.get(self.url)
        self.driver.find_element('id', 'username').send_keys('admin')  # 输入账号
        self.driver.find_element('id', 'username').send_keys('123456')  # 输入密码
        self.driver.find_element('id', 'verification').send_keys('9988')  # 输入验证码
        self.driver.find_element('xpath', '//input[@value="Login"]').click()  # 点击登录
        loginname = self.driver.find_element('id', 'loginName').text  # 登录进去显示账号名
        self.assertEqual('admin', loginname)  # 判断是否登录成功

    # 登录失败的测试数据组合
    @data(*TestData.get_data_from_file('data.txt'))  # 这里的*号是解包的意思,将testdata列表解包出5个元组,然后穿给@data,如果不加*,就是将testdata列表值整个传给@data
    @unpack  # 解包,将data里面的5种组合拆开成username, password, verification, errmsg参数传进下面函数中
    # 登录失败的测试用例,执行5次
    def testLoginFailed(self, username, password, verification, err):
        self.driver.get(self.url)
        self.driver.find_element('id', 'username').clear()
        self.driver.find_element('id', 'username').send_keys(username)  # 输入账号
        self.driver.find_element('id', 'username').clear()
        self.driver.find_element('id', 'username').send_keys(password)  # 输入密码
        self.driver.find_element('id', 'verification').clear()
        self.driver.find_element('id', 'verification').send_keys(verification)  # 输入验证码
        self.driver.find_element('xpath', '//input[@value="Login"]').click()  # 点击登录
        errmsg = self.driver.find_element('xpath', '//div[contains(text(), "{}")').format(err).text  # 登录失败显示的提示框
        self.assertEqual(errmsg, err)  # 判断是登录失败提示信息是否正确

6.断言失败截图

当预期结果与实际结果不一样的时候,进行截图,能够更清晰的了解测试用例执行失败的情况。断言失败之后的代码就不执行了,所以在断言后面加截图程序是不可行的。

1.通过异常处理的方式实现

try except

from selenium import webdriver
import unittest


class LoginTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.url = ''

    def tearDown(self) -> None:
        self.driver.quit()
        
    def test_demo(self):
        try:
            self.assertEqual(1, 2)
        except AssertionError as e:
            import os, time
            day = time.strftime('%Y%m%d', time.localtime(time.time()))  # 获取当前时间
            screenshot_path = os.getcwd() + r'\reports\screenshot_%s' % day  # getcwd()表示获取项目的路径,该句是存放截图的位置和命名方式
            if not os.path.exists(screenshot_path):  # 如果文件路径不存在
                os.makedirs(screenshot_path)  # 创建文件路径

            tm = time.strftime('%H%M%S', time.localtime(time.time()))
            self.driver.get_screenshot_as_file(screenshot_path + '\\{}_{}.png'.format('screen_shot', tm))
            raise e  # 将异常继续抛出给unittest,必须写


if __name__ == '__main__':
    test = unittest.defaultTestLoader.loadTestsFromTestCase(LoginTest)
    suite = unittest.TestSuite(test)
    unittest.TextTestRunner().run(suite)

缺点:自动化测试项目中断言会很多,每个断言都写这么多代码比较麻烦。

2.通过装饰器的方式实现(最常用)

from selenium import webdriver
import unittest


class LoginTest(unittest.TestCase):

    def setUp(self) -> None:
        self.driver = webdriver.Chrome()
        self.url = ''

    def tearDown(self) -> None:
        self.driver.quit()
        
    # 装饰器,这是一个模板,其他也可以直接用
    def addpic(func):
        def wrapper(self, *args, **kwargs):
            try:
                func(self, *args, **kwargs)
            except AssertionError as e:
                import os, time
                day = time.strftime('%Y%m%d', time.localtime(time.time()))  # 获取当前时间
                screenshot_path = os.getcwd() + r'\reports\screenshot_%s' % day  # getcwd()表示获取项目的路径,该句是存放截图的位置和命名方式
                if not os.path.exists(screenshot_path):  # 如果文件路径不存在
                    os.makedirs(screenshot_path)  # 创建文件路径

                tm = time.strftime('%H%M%S', time.localtime(time.time()))
                self.driver.get_screenshot_as_file(screenshot_path + '\\{}_{}.png'.format('screen_shot', tm))
                raise e  # 将异常继续抛出给unittest,必须写
        return wrapper()
        
    @addpic  # 断言失败会生成截图,断言成功不会生成截图
    def test_demo(self):
        self.assertEqual(1, 2)
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值