HttpRunner3.X学习笔记(8)- 参数关联与传递

前言

众所周知,在接口测试工作中,接口之间参数的关联是再常见不过的应用,Httprunner3.x在这一块的支持上用法也是相当丰富,下面结合实例简单记录一下使用技巧

变量的提取与传递

由于HttprunnerStep支持两个方式。一个是RunRequest(),即是发送一个API请求,一个是RunTestCase(),是用例中调用另一个用例。

  • 参数提取:extract()
  • 参数导出: export()

RunRequest类型的Step

在用例步骤中如果类型是RunRequest,那么只用提取变量,不需要导出就可以向下步骤中传递变量,看例子

from httprunner import HttpRunner, Config, Step, RunRequest
from utils.FakerData import *
from api.dlvopenapi import unified_order, order_query
from utils.SigleData import FakerSingleData

class TestCaseDlvOpenApiSuccess(HttpRunner):
    config = (
        Config("发放成功的测试案例")
            .variables(
            **{
                "appId": 200803014515363414,
                "accountName": get_name(),
                "idCard": "420624199109037552",
                "amount": 0.02,
                "signType":"rsa2",
                "version":"1.0",
                "accountNo": 6225380092315252,
                "phone":get_phone(),
                "serviceCompanyId": "10000032",
                "key": "MIICeAIBADANBgkqhkiG9w0BAQEFAASCAmIwggJeAgEAAoGBAOI4fc5POr0cZmyyGuwAonQGNVUvEd8KWtYk2ozNM1S6la7BK7DwWyk6kI10/NsY+Wues+C0fN86Ol0uckTF+s4NqMqw/W8PZ0HOV0mW1Ew/EV+HT1jRnbYviTRNWx9zS+xRjiDam3VdbElSwPQQulefAlQctcxsT1nPGSbWxvFpAgMBAAECgYEAnG6iGHNTVAh6j3mOAlrh+8d7Q9+bxRd8/w5XDvyrHVE1RrYPx3g+IcF8ykT2wW+Asrn4+07z9s1mJJ+EpygcqOmWtrExQlHjYcAn+27usxhTjWtNZ7AQoF0O0zIIAb8H2dm6Sin2fwvkZORaUy9gMlM7FW/LA48l49ptvs4aTZ0CQQDx5ezsesL8zA3SR+lr8xLTSjJokUOrK5wbYj+SGe+jgkUDxNU53zQXjTKM99rR+4cwomKbMFqGhZzDw1rEmccrAkEA72iYnXAKFHX05TB8YNkcxouwkFpi3rcD7m8LjEcXT/quNvsrcHfJB4iw63H2TwXvUQ4SJrxzjUMwSNkB19zfuwJBAOu6/0ns0Dv+trFndufF92CEe98/QMx8MSLWedDtCYU0HAFyPcCp7V/OL6cEmu/qyHHyrVlCo9VYO87if3/7xAUCQBi98Y/LxW7p5d5NzXzg00V9qEiy3qbvuRtKJKJhsnoUiS6rdIjSCFeb+9TJWVA/Z8UztBKGxVZjDDlrG/KoJAMCQQCRnIyn35lHhj5iXyB2Ofo5wHI45NvPpj7ut3rec58fMLTs3yw5w9OWnOSUCnbZ3gTQHeu4/3Ebu4ozzw4bi2iK"
            }
        )
            .base_url("http://192.168.1.92:11958")
            .verify(False)
            .export( *["reqNo", "outOrderNo"])
    )

    teststeps = [
        Step(
            RunRequest("发放收单成功").
                setup_hook('${setup_request($request,$key)}')
                .with_variables(
                **{
                    "attach": "附件",
                    "bank": "ICBC1",
                    "batchNo": "20200801",
                    "depositBank": "工商",
                    "extInfo": "API001-商户004-攻城狮",
                    "foreignNationality": "CN",
                    "memo": "API_AUTO1",
                    "outOrderNo": get_outOrderNo(),
                    "personalIncomeTax": "0",
                    "phone": "$phone",
                    "postId": "10468",
                    "postName": "tester",
                    "serviceFee": "0",
                    "serviceType": "10002",
                    "shouldAmount": "0",
                    "totalFee": "0",
                    "nonce": get_nonce(),
                    "timestamp": get_current_time()}
            )
                .post(unified_order.UnifiedOrderApi.path)
                .with_json(unified_order.UnifiedOrderApi.json)
                .teardown_hook('${polling_assert(state_,30)}')
                .extract()
                .with_jmespath("body.data.reqNo", "reqNo")
                .with_jmespath("body.data.outOrderNo", "outOrderNo")
                .validate()
                .assert_equal("status_code", 200)
                .assert_equal("body.code", "0000")
                .assert_equal("body.msg", "处理成功")
                .assert_equal('${db_filed_states_with_parm(state_)}', 30)
                .assert_equal('${db_filed_states_with_parm(balance_state_)}', 39)
                .assert_equal('${db_filed_states_with_parm(channel_balance_state_)}', 39)


        ),

        Step(
            RunRequest("发放订单查询")
                .setup_hook('${setup_request($request,$key)}')
                .with_variables(**{
                "method": "ayg.salary.payQuery",
                "nonce": get_nonce(),
                "timestamp": get_current_time(),

            })
                .post(order_query.OrderQueryApi.path)
                .with_json(order_query.OrderQueryApi.json)
                .validate()
                .assert_equal("status_code", 200)
                .assert_equal("body.code", "0000")
                .assert_equal("body.data.code", "30")
                .assert_equal("body.data.msg", "支付成功")
        )
    ]


if __name__ == "__main__":
    TestCaseDlvOpenApiSuccess().test_start()


如上例子:

  1. Step1中通过extract()提取了两个变量,分别是reqNo,outOrderNo,此时是不需要声明方法export()导出的。

                .extract()
                .with_jmespath("body.data.reqNo", "reqNo")
                .with_jmespath("body.data.outOrderNo", "outOrderNo")
    
  2. Step2中直接通过$reqNo,$outOrderNo引用导出的变量,即可完成参数的传递和关联

RunTestCae类型的Step

在一个测试用例中的步骤中如果调用了另一个测试用例,则需要手动导出变量,导出变量的方式两种,

  1. 在被调用用例的全局Config中导出
  2. 在调用用例的时候的步骤中导出

看例子

  • 被调用的Login用例
from api.common.login import Login
from httprunner import HttpRunner, Config, Step, RunRequest


class TestProviderLogin(HttpRunner):
    config = (
        Config("admin端登录成功")
            .base_url("https://p-test92.aiyuangong.com")
            .variables(**{
            "loginName": "console-provider",
            "password": "123456"
        })
            .verify(False)
        .export(*["userid","access_token"])
    )
    teststeps = [
        Step(
            RunRequest("登录请求")
                .post(Login.path).with_json(Login.json)
                .extract()
                .with_jmespath("body.data.userProfiles[0].id", "userid")
                .with_jmespath("body.data.access_token", "access_token")
                .validate()
                .assert_equal("body.code", 200)
        )
    ]


if __name__ == '__main__':
    TestProviderLogin().test_start()

  1. 变量的提取和导出分两个步骤,先提取

                .extract()
                .with_jmespath("body.data.userProfiles[0].id", "userid")
                .with_jmespath("body.data.access_token", "access_token")
    

    再导出

        Config("admin端登录成功")
            .base_url("https://p-test92.aiyuangong.com")
            .variables(**{
            "loginName": "console-provider",
            "password": "123456"
        })
            .verify(False)
        .export(*["userid","access_token"])
    
  2. 如果在此处导出了变量,那么在后续引用到该用例的用例的中,就不需要再在调用后导出了

  3. 如果选择在此用例中只提取,不导出,在用例调用的时候就必须进行导出操作

from httprunner import HttpRunner, RunRequest, RunTestCase,Step, Config
from api.common.login import Login
from api.common.upload import Upload
from testcases.common.p_login_success_test import TestProviderLogin


class TestUploadFileAndGetDownloadCode(HttpRunner):
    config = (
        Config("上传文件获取downloadcode")
            .base_url("https://p-test92.aiyuangong.com")
            .variables(**{
            "loginName": "console-provider",
            "password": "123456",
            "fileName_": "pkq.jpeg",
            "file_": "file/pkq.jpeg"
        })


    )
    teststeps = [
      Step(
        RunTestCase("登录后获取token")
          .call(TestProviderLogin)
          .export(*["userid","access_token"])

      ),

      Step(
        RunRequest("上传文件")
        .post(Upload.path)
        .with_headers(**{
          "Cookie": "x - sec - profile =$userid;AYG_SESSIONID =$access_token"
        })
        .upload(**Upload.file)
        .extract()
        .with_jmespath("body.data.downloadCode","downloadCode")
        .with_jmespath("body.data.fileName","fileName")
        .with_jmespath("body.data.referID","referID")
        .validate()
        .assert_equal("body.code",200)
      )

    ]

if __name__ == '__main__':
    TestUploadFileAndGetDownloadCode().test_start()

如上,Step1中调用了Login用例,而后进行了导出操作.export(*["userid","access_token"]),在Step2中就进行变量的关联传递了

思考一下

两个地方的导出最佳实践是什么?

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