pytest用例的执行顺序

Pytest执行的顺序

当pytest运行测试函数时,它会查看该测试函数中的参数,然后搜索与这些参数具有相同名称的fixture。一旦pytest找到这些对象,它就会运行这些fixture

影响执行顺序的因素

       1. fixture方法的scope:fixture的使用范围

       2. dependencies:可能会存在fixture请求了别的fixture,所以产生了依赖关系

       3. autouse:如果多个fixture,其中一个autouse=True,那么这个fixture先执行

执行顺序:

单个fixture,先执行这个fixture,再执行测试用例

多个fixture,就要确定fixture的执行顺序

       1. fixture设置了autouse=True,这个autouse的fixture函数会比请求的其他fixture都要先执行。

       2. Scope使用范围大的fixture先执行:"session"> "package"> "module"> "class">"function"

       3. 相同顺序的fixture基于依赖项执行:当一个fixture函数请另一个fixture函数,另一个会先执行,参考:https://www.cnblogs.com/pingguo-softwaretesting/p/14698711.html

fixture引用方式

       1. 一个fixture函数:A  请求另一个fixture:B,另一个B会先执行

       2. 为了控制B的fixture先执行,也可以通过autouse设置,将autouse设置成True

       3. 第三种方法,给先执行的fixture通过参数化,传入参数,

       调用方式

              1.  通过将fixture声明为参数来请求fixture,注意fixture名字不能写错

              2.  通过pytest.mark.usefixtures,将fixture声明作为参数传入测试用例

!!!pytest收集到测试用例的执行顺序

       1.  判断一下当前测试函数,是否有使用参数化,参数化里面有几组的测试数据,有几组的数据,决定了这条测试用例执行多少次

       2.  有使用参数化,参数化有两组的数据,分别执行两次的测试用例

       3.  测试用例执行前,会先查看该测试函数中的参数,然后搜索与函数名称一样的fixture,找到了这些fixture,先执行fixture,然后回到测试用例,继续执行

       运行 》 查找测试用例 》 找到测试用例 》 判断当前有几组参数化数据 》 开始执行 》 查看测试函数中的参数,与conftest中的fixture的名称是否有一样的 》 执行fixture 》 回到测试用例,开始执行测试用例

实际项目:

传入测试用例的数据:

sucess_data_bak = [{"user_id": "${user_id}", "username": "${login_user}", "password": "123", "sku_code": "vivo00100015",

                   "check": "第二个用户登录"},

                   {"user_id": "${user_id}", "username": "${login_user}", "password": "123", "case": "正常登录",

                   "check": "第一个用户登录"}]

  

测试用例

@pytest.mark.usefixtures("open_erp_url")    # 使用fixture方法:open_erp_url

@pytest.mark.usefixtures("preset_users_conditions")  # 使用fixture方法:preset_users_conditions

# 将参数传递给fixtures,命名为test_data,要使用这个参数的fixture的方法,传入的参数名称要和test_data一样

@pytest.mark.parametrize("test_data", login_data.sucess_data)

@pytest.mark.testlogin

# open_erp_url,用到几个fixtures、给fixtures传参:就要传几个fixtures进来测试用例

def test_login(self, open_erp_url, preset_users_conditions, test_data):

    '''
    :param open_erp_url: 前置条件打开浏览器,返回driver

    :param preset_users_conditions: 用户前置条件,包含查询用户的数据,和更新用户的数据

    :param sucess_data: 参数化的测试用例的数据

    :return:

    '''

    # 调试fixtures的登录方法

    data = preset_users_conditions

    print("开始执行登录成功的前置条件")

    loginpage(open_erp_url).Login(preset_users_conditions["username"], preset_users_conditions["password"])

    time.sleep(2)

  

测试用例使用了参数化

@pytest.mark.parametrize("test_data", login_data.sucess_data),两组数据,测试用例要执行两遍,将这个参数传到了fixture里面,根据参数化,先执行带参数化的fixture

测试用例执行前,查看测试函数中的参数,与conftest中的fixture的名称是否有一样,一样的先执行fixture

@pytest.mark.parametrize("test_data", login_data.sucess_data)

@pytest.mark.testlogin

# open_erp_url,用到几个fixtures、给fixtures传参:就要传几个fixtures进来测试用例

def test_login(self, open_erp_url, preset_users_conditions, test_data):
    loginpage(open_erp_url).Login(preset_users_conditions["username"], preset_users_conditions["password"])
 
 
 
@pytest.fixture(scope="function") def preset_users_conditions(test_data): ''' 用户的前置条件,包含传进来的test_case的数据,判断查询什么样的用户登录、以及根据user_id更新用户登录的密码、已经用户的角色,分配给某个角色 :param test_data: test_login.py测试类中测试用例,通过参数化,命名的参数化的数据,名称要和参数名声明的保持一致, 如:@pytest.mark.parametrize("test_data", login_data.sucess_data) :return: ''' # 根据测试用例的数据,判断要查询的是什么用户,用于登录中台 res_data = datamanage.dynamic_replace_data(testdata=test_data) # 更新用户的密码为123 update_user_pwd(res_data["user_id"]) yield res_data

执行完fixture:preset_users_conditions,接着执行open_erp_url

@pytest.fixture(scope="function")

def open_erp_url():

    '''

    执行登录操作前,先做打开浏览器,设置成最大化,然后再做登录操作

    :return:  driver:将当前打开浏览器的driver,传递给后面使用

    '''

    MyLog().info("执行前置方法_open_erp_url,打开浏览器")

    driver = webdriver.Chrome()

    driver.maximize_window()

    MyLog().info("open_erp_url_打开登录页面")

    driver.get(DoInfo.host)   # url登录的地址

    yield driver   # 返回driver给后面的其他操作使用

    driver.quit()  # 后置条件:测试用例执行完毕,关闭浏览器

Fixture执行完毕之后,回到测试用例, 执行测试用例

loginpage(open_erp_url).Login(preset_users_conditions["username"], preset_users_conditions["password"])