一、Fixture介绍
Fixture是pytest精髓所在,就像unittest中的setup和teardown一样,但相对之下它的功能更加强大和灵活。
二、Fixture的作用
1.做测试前后的初始化设置,如测试数据准备,链接数据库,打开浏览器等这些操作都可以使用fixture来实现
2.测试用例的前置条件可以使用fixture实现
3.支持经典的xunit fixture ,像unittest使用的setup和teardown
4.fixture可以实现unittest不能实现的功能,比如unittest中的测试用例和测试用例之间是无法传递参数和数据的,但是fixture却可以解决这个问题
三、Fixture定义及调用
fixture通过@pytest.fixture()装饰器装饰一个函数,那么这个函数就是一个fixture,看个实例:
@pytest.fixture() def manage_browser(): driver = webdriver.Chrome() # 前置条件 return driver print("后置条件") # 后置条件 @pytest.mark.fixe def test_baidu(manage_browser): driver = manage_browser driver.get("https://www.baidu.com/")
声明的测试夹具函数名称,作为参数传入其他函数,即可完成调用。可以传多个fixture,按先后顺序执行。
上面的测试夹具中只用return返回初始化浏览器,return后面的后置条件不会执行,因此需要将return改成yield,后置条件就会被调用执行。
Yield是Python中的一个关键字,表示生成器。测试夹具的前置条件遇到yield时返回一个结果,然后将测试夹具挂起,转而去执行测试用例,最后回来测试夹具中执行后置条件。Return和yield后面可以不写任何返回值,表示空没有数据返回。Yield使用示例如下:
import pytest from selenium import webdriver @pytest.fixture() def manage_browser(): # 前置条件 driver = webdriver.Chrome() yield driver # yield,返回函数值,可以继续执行下面的代码 # 后置条件 driver.quit() @pytest.mark.fix def test_baidu(manage_browser): driver = manage_browser # 这里是参数接收,不能打括号 driver.get("https://www.baidu.com/")
四、Pytest中fixture前置后置条件
Pytest框架中将前置条件,后置条件单独放在fixture的函数中,所有的用例调用测试夹具的前后置条件。Web自动化的前后置条件如下:
@pytest.fixture() def manage_browser(): """初始化浏览器""" driver = webdriver.Chrome() # 设置隐式等待 driver.implicitly_wait(20) # 最大化浏览器 driver.maximize_window() # 初始化要用到的页面 login_page = LoginPage(driver) # 初始化首页 index_page = IndexPage(driver) yield driver, login_page, index_page # 后置条件 driver.quit() @pytest.fixture() # 可以同时创建多个测试夹具 def login(): """登录的夹具""" manage_browser() pass
要使用测试夹具,在测试用例类中的test方法中传入夹具函数名(例如上面函数的manage_browser),且框架中所有的测试用例文件中不需要导入夹具函数名,是直接使用的。如下代码:
class TestLogin: @pytest.mark.parametrize("test_info", test_data_error) def test_01_login_error(self, test_info, manage_browser): """登录时,手机号为空""" # 拆开元祖中的参数 driver, login_page, index_page = manage_browser # 第一步:登录 login_page.login(test_info["mobile"], test_info["pwd"]) # 获取实际结果(封装以后执行的函数或者方法) actual = login_page.get_error_msg() # 第二步:获取预期结果 test_info["expected"] expected = test_info["expected"] # 第三步:断言 # self.assertEqual(expected, actual) unitte中的断言方式 assert expected == actual
测试用例类中使用测试夹具,以及数据传递的方式:
注意:Pytest中的fixture测试夹具单独放在conftest.py文件中,且conftest.py文件需要与pytest启动文件放在同一级目录。
五、Fixture作用域
Unittest框架中setup的作用是每条测试用例执行之前都会执行一次,setupclass的作用是每个测试用例类执行之前都会执行一次。
pytest的fixture同样有这样的作用域,且使用更广泛更灵活。
关键代码:@pytest.fixture(scope='作用范围'),参数如下:
???? function:默认作用域,每个测试用例都运行一次
???? class:每个测试类只执行一次
???? module:每个模块只执行一次(模块:一个.py文件)
???? package:每个python包只执行一次
???? session:整个会话只执行一次,即运行项目时整个过程只执行一次
Fixture后面的括号不加任何参数,就代表默认作用域,与function作用一样。
示例一:class --每个测试类只执行一次
import pytest from selenium import webdriver test_data = [1, 2, 3] @pytest.fixture(scope="class") def manage_browser(): """初始化浏览器""" driver = webdriver.Chrome() yield driver # 后置条件 driver.quit() class TestHelloWord: @pytest.mark.parametrize("test_info", test_data) def test_helloword(self, test_info, manage_browser): pass def test_hello(self, manage_browser): pass
上面的示例TestHelloWord虽然有4条测试用,但是@pytest.fixture(scope="class") 申明了每个测试类只执行一次,所以整个过程只打开了一次浏览器。如果一个.py文件中有2个类,就会打开2次浏览器。
示例一:module --每个模块只执行一次
import pytest from selenium import webdriver test_data = [1, 2, 3] @pytest.fixture(scope="module") def manage_browser(): """初始化浏览器""" driver = webdriver.Chrome() yield driver # 后置条件 driver.quit() class TestHelloWork: @pytest.mark.parametrize("test_info", test_data) def test_helloword(self, test_info, manage_browser): pass def test_hello(self, manage_browser): pass class TestModule: @pytest.mark.parametrize("test_info_2", test_data) def test_module(self, test_info_2, manage_browser): pass def test_module2(self, manage_browser): pass
上面的示例中有两个类,8条测试用例,但是测试夹具作用域设为module --每个模块只执行一次,所以整个过程只打开一次浏览器。
特殊方式:fixture里面有个参数autouse(自动使用的意思),默认是False,当设置为True时,用例就会自动调用该fixture功能,这样的话写用例时就不用每次都去传参了。
六、多个测试夹具
当有多个测试夹具,可以设置不同的作用域,测试用例方法后面可以传需要的测试夹具,也可以传入全部测试夹具
test_data = [1, 2, 3] @pytest.fixture(scope="function") def manage_browser(): """初始化浏览器""" driver = webdriver.Chrome() yield driver print("后置执行") # 后置条件 driver.quit() @pytest.fixture(scope="class") def delete_cookie(): print("delete cookie before") yield print("delete cookie after") class TestHelloWork: @pytest.mark.parametrize("test_info", test_data) def test_helloword(self, test_info, manage_browser, delete_cookie): pass