框架其实是开发大佬定制研发的应用骨架,一个半成品,它对基础的代码进行了封装并且提供一些API接口,其他开发者只需要调用封装好的API接口即可,可以省去很多代码编写,从而提高工作效率。
自动化测试leader为了对一个系统做自动化测试而封装的一个代码主骨架,其他的自动化测试工程师只需要去调用这个骨架里面的方法就可以实施自动化测试,这个代码骨架就叫自动化框架。
1.提高测试效率,降低维护成本;
2.减少人工干预脚本的因素;
3.增加代码的可重用率。
单元测试:指的是对我们程序的最小单元(方法)进行测试。
unittest单元测试框架 是 自动化测试框架 的重要组成部分之一;
POM,ddt数据驱动,全局配置文件封装,selenium二次封装,日志监控,断言,邮件发送等都是自动化测试框架的重要组成部分之一。
基于python语言:unittest和pytest
基于Java语言:Junit和testing
unittest:提供了testcase测试用例、testsuite测试套件、testfixture测试固件或夹具、testloader测试加载器、testrunner测试运行器。必须遵守以下规则:
(1)测试文件必须先导入import unittest
(2)测试类必须继承 unittest.TestCase
(3)测试方法必须以 test 开头
pytest:它是python的第三方测试框架,基于unittest的扩展框架。必须遵守以下规则:
(1)测试文件名必须以 test_ 开头或者是 _test 结尾
(2)测试类命名必须以 Test 开头
(3)测试方法必须 test 开头
unittest | 解释 |
---|---|
setUp / tearDown | 在每个用例之前或之后运行一次【一般用于打开浏览器,加载网页/关闭网页】 |
setUpClass / tearDownClass | 在每个类运行之前或之后运行一次【一般用于创建数据库连接/关闭数据库连接,创建日志对象/销毁日志】 |
setUpModule / tearDownModule | 在每个模块之前或之后执行一次 |
pytest | 解释 |
---|---|
setup_method / teardown_mothod setup / teardown | 在每个方法之前或之后执行一次 |
setup_function / teardown_function | 在每个函数之前或之后执行一次 |
setup_class / teardown_class | 在每个类之前或之后执行一次 |
setup_module / teardown_module | 在每个模块之前或之后执行一次 |
另外,还可以在函数之前加@pytest.fixture().
unittest:assertTure,assertEqual,assertIn
pytest:assert
unittest:htmltestrunner
pytest:【插件】pytest-HTML,allure
unittest:没有
pytest:pytest-rerunfailures插件
unittest:ddt
pytest:@pytest.mark.parametrize 装饰器
unittest:默认执行所有,也可以通过 testsuite 来执行部分用例,或者 -k 参数
pytest:@pytest.mark
测试发现:从多个py文件中收集并且加载测试用例
测试执行:将测试用例按照一定的顺序和条件去执行,并且生成结果
测试判断:通过断言去判断结果是否正确
测试报告:统计测试进度、通过率,生成报告
import unittest
class MarketLogin(unittest.TestCase):
#测试用例
def test01_admin(self):
print("测试test01.admin")
if __name__ == '__main__':
print("---------------------------")
unittest.main()
python -m unittest -v market_login.MarketLogin
python -m 以脚本的方式去运行一个模块
unittest -v --verbose 更加详细地输出结果
market_login.MarketLogin 模块名.类名.方法名
-k匹配模式:
通配符:-k *_user 表示必须以_user结尾
字符串:-k user 表示只要包含user即可
若只想执行部分测试用例,则可以用testsuite
import unittest
class MarketLogin(unittest.TestCase):
#测试用例
def test01_admin(self):
print("测试test01.admin")
def test01_user(self):
print("测试test01.user")
if __name__ == '__main__':
print("测试结果")
suite = unittest.TestSuite()
suite.addTest(MarketLogin("test01_admin"))
unittest.TextTestRunner().run(suite) #替换成 unittest.main(suite) 也可以
import unittest
class MarketLogin(unittest.TestCase):
#测试用例
def test01_admin(self):
print("测试test01.admin")
def test01_user(self):
print("测试test01.user")
if __name__ == '__main__':
print("测试结果")
suite = unittest.TestSuite()
testcases = [MarketLogin("test01_admin"), MarketLogin("test01_user")]
suite.addTests(testcases)
unittest.TextTestRunner().run(suite)
import os
import unittest
class MarketLogin(unittest.TestCase):
#测试用例
def test01_admin(self):
print("测试test01.admin")
def test01_user(self):
print("测试test01.user")
if __name__ == '__main__':
print("测试结果")
suite = unittest.TestSuite()
testcases = unittest.defaultTestLoader.discover(start_dir=os.getcwd(),pattern='*.py')
suite.addTests(testcases)
unittest.TextTestRunner().run(suite)
import unittest
class MarketLogin(unittest.TestCase):
@classmethod
def setUpClass(cls) -> None:
print("setUpClass每个类之前执行一次,如:创建数据库,生成日志对象")
def setUp(self) -> None:
print("\nsetUp在测试用例执行之前的准备工作,如:打开浏览器,加载网页")
def test01_admin(self):
print("\n测试test01.admin")
def test01_user(self):
print("\n测试test01.user")
def tearDown(self) -> None:
print("\ntearDown在测试用例执行之后的收尾工作,如:关闭浏览器")
@classmethod
def tearDownClass(cls) -> None:
print("\ntearDownClass每个类之后执行一次,如:关闭数据库,销毁日志对象")
if __name__ == '__main__':
unittest.main()
按ASCII码的规则:【0-9 A-Z a-z】A=65 a=97
module=‘main’, 测试用例所在的路径,__main__表示当前模块
defaultTest=None, 待测用例的名称,默认是所有
argv=None,接收外部的参数
testRunner=None, 测试运行器,TextTestRunner
testLoader=loader.defaultTestLoader,指定使用默认的测试用例加载器
exit=True, 是否在测试完成之后结束python的程序
verbosity=1,类似于命令行-v,更加详细地输出结果
failfast=None,失败后是否终止测试
catchbreak=None,捕获
buffer=None, 输出流
warnings=None, *, 警告信息
tb_locals=False
import unittest
from myunit import MyUnit
class MarketLogin(MyUnit):
@unittest.skip("已测试")
def test01_user(self):
print("\n测试test01.user")
def test02_user(self):
print("\n测试test02.user")
import unittest
from myunit import MyUnit
class MarketLogin(MyUnit):
age = 100
@unittest.skip("已测试")
def test01_user(self):
print("\n测试test01.user")
@unittest.skipIf(age>50 and age<=100,"如果在这个范围就忽略")
def test02_user(self):
print("\n测试test02.user")
import unittest
from myunit import MyUnit
class MarketLogin(MyUnit):
age = 100
@unittest.skip("已测试")
def test01_user(self):
print("\n测试test01.user")
@unittest.skipIf(age>50 and age<=100, "如果在这个范围就忽略")
def test02_user(self):
print("\n测试test02.user")
@unittest.skipUnless(age<18, "除..以外忽略,也就是说,满足前面的条件就不忽略,不满足前面的条件就不忽略,100不满足则忽略")
def test03_user(self):
print("\n测试test03.user")
数据文件中有11组数据,代码只需要一份,代码数据分离,解耦合(将依赖关系降到最低)
数据驱动
关键字驱动 (核心:将业务逻辑封装成关键字login,后续只需要调用login)
混合驱动 (关键字驱动+数据驱动):市场主流
行为驱动测试:Lettuce
data driver test:数据驱动测试。
特点:可以完美应用于unittest框架实现数据驱动
ddt是通过装饰器的形式来调用的
装饰器:完成一种特定功能的函数(事物),在python里面装饰器是以@开头,并且装饰器有两种 —— 类装饰器和函数装饰器
ddt里面有哪些装饰器?
@ddt (申明当前类使用ddt框架)
@data (函数装饰器,用于给测试用例传递数据)
使用@data传递多个值时,传递几个值,用例就会执行几次
@unpack (将传输的数据包解包),一般作用于元组tuple和列表list
@file_data (函数装饰器,可直接读取 yaml / json 文件)
如何安装ddt?
点击 file —— settings —— project:项目名 —— ‘+’ —— 在输入框中输入ddt —— 选中ddt并点击左下角的 “install package”,提示installed successfully即表示安装成功
创建两个py文件:market_login.py 和 runpy.py
import unittest
from ddt import ddt, data
@ddt
class MarketLogin(unittest.TestCase):
@data(10)
def test01_user(self, args):
print(args)
def test02_user(self):
print("\n测试test02.user")
def test03_user(self):
print("\n测试test03.user")
import os
import unittest
if __name__ == '__main__':
suite = unittest.TestSuite()
testcases = unittest.defaultTestLoader.discover(os.getcwd(), '*.py')
suite.addTests(testcases)
unittest.TextTestRunner(verbosity=2).run(suite)
如果是数字或者字符串,则不需要@unpack
如果是元组或列表,那么可以用@unpack,不过参数的个数必须和解完包后的值的个数一样
如果是集合则无法解包
如果是字典,可以用@unpack,不过参数的名字和个数必须和字典的键保持一致