• 测试开发 | Java 接口自动化测试首选方案:REST Assured 实践


    在这里插入图片描述

    1 . 初识 REST Assured

    在 REST Assured 的官方 GitHub 上有这样一句简短的描述: Java DSL for easy testing of REST services 简约的 REST 服务测试 Java DSL

    1.1 优点:

    REST Assured 官方的 README 第一句话对进行了一个优点的概述,总的意思表达的就是简单好用。那么 REST Assured 有哪些优点,又该如何使用呢?

    图片

    用 Java 做接口自动化测试首选 REST Assured,具体原因如下:

    开源

    简约的接口测试 DSL

    支持 xml json 的结构化解析

    支持 xpath jsonpath gpath 等多种解析方式

    对 spring 的支持比较全面

    功能很齐全,部分我自己也还没有具体用到,了解到了方向,需要时随时查找学习

    2. 如何使用

    添加 maven 依赖

    1. io.rest-assured
    2. rest-assured
    3. 4.0.0
    4. test

    2.1 基本三步曲

    我们对接口进行测试一般由三步曲:传参、发请求、响应结果断言,REST Assured给我们提供了清晰的三步曲,以given、when、then的结构来实现,基本写法如下:

    1. //使用参数
    2. given().
    3. param("key1", "value1").
    4. param("key2", "value2").
    5. when().
    6. post("/somewhere").
    7. then().
    8. body(containsString("OK"))
    9. //使用X-Path (XML only)
    10. given().
    11. params("firstName", "John", "lastName", "Doe").
    12. when().
    13. post("/greetMe").
    14. then().
    15. body(hasXPath("/greeting/firstName[text()='John']"))

    2.2 分步拆解

    前提:现有一个post请求的登录接口。

    http://47.103.xxx.133/auth/oauth/token,

    请求体body如下

    1. {
    2. "password": "elcrD28ZSLLtR0VLs/jERA\u003d\u003d\n",
    3. "grant_type": "password",
    4. "scope": "server",
    5. "userType": 1,
    6. "username": "xxx"
    7. }

    Request Header 如下:

    1. Headers: Authorization=Basic c3lzdGVtxxxRlbQ==
    2. Host=47.103.xxx.133
    3. Accept=*/*
    4. Content-Type=application/json; charset=ISO-8859-1

    分步拆解一

    Givern

    我们发送请求经常需要带有参数,使用 given() 就可以实现,当时当我们使用 given() 的时候发现其中有很多传参方法如下:

    图片

    没错,在传参的方法中包含了 param、pathParam、queryParam 和 formParam,下面来研究下这几个传参方法的区别

    param

    通常我们都会使用 given().param 方法来传参,REST Assured 会根据 HTTP 方法自动尝试确定哪种参数类型(即查询或表单参数),如果是 GET,则查询参数将自动使用,如果使用 POST,则将使用表单参数;

    queryParam 和 formParam

    有时候在 PUT 或 POST 请求中,需要区分查询参数和表单参数时,就需要使用queryParam 和 formParam 方法了,具体写法如下:

    1. given().
    2. formParam("formParamName", "value1").
    3. queryParam("queryParamName", "value2").
    4. when().
    5. post("/something")

    pathParam

    使用given时指定请求路径的参数,这个方法很少用到,或者说我本人几乎没用到过(可能我的修行还不够,踩坑还太少~);具体写法如下:

    1. given().
    2. pathParam("OAuth", "oauth").
    3. pathParam("accessToken", "token").
    4. when().
    5. post("/auth/{OAuth}/{accessToken}").
    6. then().
    7. ..

    header/headers

    经常还需要在请求头中带入参数,这个时候就可以使用header或headers方法,写法如下:

    1. given()
    2. .header("Authorization","Basic c3lzdGVtOxxxbQ==")
    3. .header("Host","47.xxx.xxx.133")

    或者用headers将多个参数写在一起:

    1. given()
    2. .headers("Authorization","Basic c3lzdGVtxxx3RlbQ==","Host","47.xxx.xxx.133")

    cookie

    有时候需要在请求中带入cookie,restassured提供了cookie方法来实现:

    1. given()
    2. .cookie("c_a","aaaaaa")
    3. .cookie("c_b","bbbbbb"). ..

    contentType

    经常还会设置contentType,最常见的就是application/json了,写法如下:

    1. given().contentType("application/json"). ..
    2. //或者
    3. given().contentType(ContentType.JSON). ..

    body

    在POST, PUT 或 DELETE请求中,我们经常还需要带上请求体body,写法如下:

    1. given().body("{\n" +
    2. "\t\"password\": \"elcrD28xxxR0VLs/jERA\\u003d\\u003d\\n\",\n" +
    3. "\t\"grant_type\": \"password\",\n" +
    4. "\t\"scope\": \"server\",\n" +
    5. "\t\"userType\": 1,\n" +
    6. "\t\"username\": \"xxx\"\n" +
    7. "}")

    也可以用request更为明确的指出是请求body:

    1. given().request().body("{\n" +
    2. "\t\"password\": \"elcrD28xxxR0VLs/jERA\\u003d\\u003d\\n\",\n" +
    3. "\t\"grant_type\": \"password\",\n" +
    4. "\t\"scope\": \"server\",\n" +
    5. "\t\"userType\": 1,\n" +
    6. "\t\"username\": \"xxx\"\n" +
    7. "}")

    没有参数

    如果我们没有参数需要传递,也可以省略掉given():

    get("/lotto").then().assertThat().body("lotto.lottoId", equalTo(5));
    

    proxy

    有时候我们需要进行接口的调试,抓包是最常用的一种方式,rest-assured 提供了 proxy 方法,可以设置代理,写法如下:

    1. given().proxy("127.0.0.1",8888). ..

    实际运行结果:

    在这里插入图片描述

    在这里插入图片描述

    分步拆解二

    When

    when主要用来触发请求,在when后面接着请求URL:

    given().when().post("http://47.103.xxx.133/auth/oauth/token"). ..
    

    前面在 given 中我们设置了很多请求参数,在 when 中也可以设置,只不过要注意的是在请求之前设置;这也比较好理解,如果再请求之后的话,参数都设置怎么发请求呢?

    1. given()
    2. .when()
    3. .contentType(ContentType.JSON)
    4. .headers("Authorization","Basic c3lzxxx3RlbQ==","Host","47.xxx.xxx.133")
    5. .request().body("{\n" +
    6. "\t\"password\": \"elcrD28ZSLLtR0VLs/jERA\\u003d\\u003d\\n\",\n" +
    7. "\t\"grant_type\": \"password\",\n" +
    8. "\t\"scope\": \"server\",\n" +
    9. "\t\"userType\": 1,\n" +
    10. "\t\"username\": \"qinzhen\"\n" +
    11. "}")
    12. .post("http://47.xxx.xxx.133/auth/oauth/token")
    13. . ..

    分步拆解三

    Then

    then后面可以跟断言,也可以获取响应值

    断言-then().body()

    then().body() 可以对响应结果进行断言,在 body 中写入断言:

    1. .. post("http://47.xxx.xxx.133/auth/oauth/token")
    2. .then().statusCode(200).body("code",equalTo(1));

    其中statusCode(200)是对状态码的断言,判断状态码是否为200; body(“code”,equalTo(1))是对返回体中的 code 进行断言,要求返回 code值为1 。

    注:这里的equalTo使用的是hamcrest断言,不了解的小伙伴可参考另外一篇文章:Junit原生断言和hamcrest断言的区别及使用

    实操演示:

    我们将上述的 given、when、then 结合起来看一下实际运行效果,这里在运行之前再提一个功能,我们可以在 when 和 then 后面加上.log().all(),这样在运行过程中就可以把请求和响应的信息都打印出来:

    在这里插入图片描述

    在这里插入图片描述

    获取响应-then().extract().body().path(“code”)

    我们可以在 then 后面利用 .extract().body() 来获取我们想要 body 的返回值,它们也可以直接接在断言后面,写法如下:

    注意这里的body() 不要和请求体body()以及断言的body()混淆了

    1. .. .then()
    2. .log().all().statusCode(200).body("code",equalTo(1))
    3. .extract().body().path("code");

    实操演示:

    演示前再来看一个新的功能,上面我们再写请求体 body 时时这样的:

    1. body("{\n" +
    2. "\t\"password\": \"elcrD28ZxxxVLs/jERA\\u003d\\u003d\\n\",\n" +
    3. "\t\"grant_type\": \"password\",\n" +
    4. "\t\"scope\": \"server\",\n" +
    5. "\t\"userType\": 1,\n" +
    6. "\t\"username\": \"qinzhen\"\n" +
    7. "}")

    看起来有点丑,改造一下;rest-assured 为我们提供了一个利用 HashMap 来创建json 文件的方法,先把要传的字段放入 hashmap 中,然后用 contentType 指明JSON 就可以了,具体写法如下:

    1. HashMap map = new HashMap();
    2. map.put("password","elcrD28ZSLLtR0VLs/jERA\u003d\u003d\n");
    3. map.put("grant_type","password");
    4. map.put("scope","server");
    5. map.put("userType",1);
    6. map.put("username","xxx");
    7. given()
    8. .headers("Authorization","Basic c3lzdGVtxxxlbQ==","Host","47.xxx.xxx.133")
    9. .contentType(JSON)
    10. .body(map). ..

    现在进行完整的请求,获取返回值 code 并打印:

    1. HashMap map = new HashMap();
    2. map.put("password","elcrD28ZSLLtR0VLs/jERA\u003d\u003d\n");
    3. map.put("grant_type","password");
    4. map.put("scope","server");
    5. map.put("userType",1);
    6. map.put("username","xxx");
    7. Integer code =
    8. given()
    9. .headers("Authorization","Basic c3lzdGVtxxxlbQ==","Host","47.xxx.xxx.133")
    10. .contentType(JSON)
    11. .body(map).
    12. when()
    13. .log().all().post("http://47.xxx.xxx.133/auth/oauth/token").
    14. then()
    15. .log().all().statusCode(200).body("code",equalTo(1))
    16. .extract().body().path("code");
    17. System.out.println("返回code的值是:"+code);

    运行结果:

    图片

    推荐学习

    关于REST Assured,这里仅仅算是初步认识。认识它的语法结构和功能,对于更多丰富的用法还需要慢慢探索研究,特别是断言的部分,是测试工程师最常用最终要的功能之一。REST Assured提供的完整断言手段,在后续文章中我们一起探讨。

    最后感谢每一个认真阅读我文章的人,礼尚往来总是要有的,虽然不是什么很值钱的东西,如果你用得到的话可以直接拿走:

    在这里插入图片描述

    这些资料,对于【软件测试】的朋友来说应该是最全面最完整的备战仓库,这个仓库也陪伴上万个测试工程师们走过最艰难的路程,希望也能帮助到你!   

  • 相关阅读:
    对话MySQL之父:一个优秀程序员可抵5个普通程序员
    STM32CUBEIDE(11)----输出PWM及修改PWM频率与占空比
    leetcode 698. 划分为k个相等的子集
    电脑开不了机用U盘重装系统Win10教程
    事务的传播机制
    NAT地址转换,路由器作为出口设备,实现负载分担
    php使用sqlServer
    使用Softing为西门子工业边缘开发的edgePlug软件简化了设备与应用程序的连接
    数据库系统原理与应用教程(001)—— MySQL 安装与配置:MySQL 软件的安装(windows 环境)
    机器学习笔记 - GluonCV:基于MXNet/PyTorch + Kinetics400 + 各种先进网络的动作识别的预训练模型
  • 原文地址:https://blog.csdn.net/qq_48811377/article/details/132759986