You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

266 lines
17 KiB
Markdown

2 years ago
# 22 | 从0到1API测试怎么做常用API测试工具简介
你好我是茹炳晟我今天分享的主题是“从0到1API测试怎么做常用API测试工具简介”。
在第11篇文章[《互联网产品的测试策略应该如何设计?》](https://time.geekbang.org/column/article/11462)中,我介绍过当今互联网产品的测试策略往往会采用菱形结构,即重量级 API 测试,轻量级 GUI 测试轻量级单元测试由此可见API测试在现今测试中的重要性不言而喻。
这篇文章是API自动化测试系列的第一篇文章我会先为你打好API测试的基础。所以我会先从0到1设计一个API测试用例通过这个测试用例你可以体会到最基本的API测试是如何进行的并介绍几款常用的API测试工具。
## API测试的基本步骤
通常来讲无论采用什么API测试工具API测试的基本步骤主要包括以下三大步骤
1. 准备测试数据这是可选步骤不一定所有API测试都需要这一步
2. 通过API测试工具发起对被测API的request
3. 验证返回结果的response。
对API的测试往往是使用API测试工具比如常见的命令行工具cURL、图形界面工具Postman或者SoapUI、API性能测试的JMeter等。
为了让你更好地理解API测试具体是怎么做的并掌握常见API测试工具的使用我会以基于主流Spring Boot框架开发的简单Restful API为例分别介绍如何使用cURL和Postman对其进行最基本的功能测试目的是让你对API测试有一个基本的感性认识。
## 基于Spring Boot构建的API
因为基于Spring Boot从0到1构建一个API并不是本文的重点为了不影响你对文章主要内容的把握我直接采用了一个预先开发好的Account API为例展开讲解。你可以从[https://github.com/SpectoLabs/spring-cloud-contract-blog](https://github.com/SpectoLabs/spring-cloud-contract-blog)下载完整的代码。
这个Account API的功能非常简单就是基于你提供的ID值创建一个Account对象并返回这个新创建Account对象。
比如如果你的请求是“account/ID008”那么返回的response就应该是“{“id”:“ID008”,“type”:“friends”,“email”:“robin@api.io”}”。
这个Account API的功能逻辑实现非常简单图1和图2列出了主要的代码逻辑。
图1中代码的第21行说明了API的endpoint以及对应的操作是GET方法第22行明确说明了GET方法具体的业务逻辑是由accountService.getById()方法实现的。
![](https://static001.geekbang.org/resource/image/22/6e/220ed0f17737529eee9c1284fba6d06e.png)
图1 RestController的实现
图2中代码的第8行实现了accountService.getById()方法具体逻辑就是返回一个以传入ID为ID的Account对象。
![](https://static001.geekbang.org/resource/image/10/fb/10a26edf4c7681a3cfbb68654a1d3bfb.png)
图2 具体业务逻辑的实现
我推荐使用IntelliJ打开这个下载的项目然后直接启动其中的account-service。启动成功后account-service会运行在本地机器的8080端口。启动成功后的界面如图3所示。
![](https://static001.geekbang.org/resource/image/c4/8e/c45ca4ea536e130374e3da4f8426718e.png)
图3 成功启动基于Spring Boot的Account API
## 使用cURL命令行工具进行测试
首先你需要下载安装cURL然后就可以通过以下命令发起Account API的调用。调用结束后的界面如图4所示。
```
curl -i -H "Accept: application/json" -X GET "http://127.0.0.1:8080/account/ID008"
```
![](https://static001.geekbang.org/resource/image/82/52/82571f2ccf198739ed4162574a851f52.png)
图4 使用cURL测试Account API
这行命令中参数的含义如下:
* 第一个参数“-i”说明需要显示response的header信息
* 第二个参数“-H”用于设定request中的header
* 第三个参数“-X”用于指定执行的方法这里使用了GET方法其他常见的方法还有POST、PUT和DELETE等如果不指定“-X”那么默认的方法就是GET。
* 最后“ [http://127.0.0.1:8080/account/ID008](http://127.0.0.1:8080/account/ID008) ”指明了被测API的endpoint以及具体的ID值是“ID008”。
当使用cURL进行API测试时常用参数还有两个
* “-d”用于设定http参数http参数可以直接加在URL的query string也可以用“-d”带入参数。参数之间可以用“&”串接,或使用多个“-d”。
* “-b”当需要传递cookie时用于指定cookie文件的路径。
需要注意的是这些参数都是大小写敏感的。
了解了这几个最常用的参数后我再来分析一些最常用的cURL命令以及使用的场景包括Session的场景和Cookie的场景。
**第一Session的场景**
如果后端工程师使用session记录使用者登入信息那么后端通常会传一个 session ID给前端。之后前端在发给后端的requests的header中就需要设置此session ID后端便会以此session ID识别出前端是属于具体哪个session此时cURL的命令行如下所示
```
curl -i -H "sessionid:XXXXXXXXXX" -X GET "http://XXX/api/demoAPI"
```
**第二Cookie的场景**
如果是使用cookie在认证成功后后端会返回cookie给前端前端可以把该cookie保存成为文件当需要再次使用该cookie时再用“-b cookie\_File” 的方式在request中植入cookie即可正常使用。具体的cURL的命令行如下所示
```
// 将cookie保存为文件
curl -i -X POST -d username=robin -d password=password123 -c ~/cookie.txt "http://XXX/auth"
// 载入cookie到request中
curl -i -H "Accept:application/json" -X GET -b ~/cookie.txt "http://XXX/api/demoAPI"
```
最后需要特别说明的是cURL只能发起API调用而其本身并不具备结果验证能力结果验证由人完成所以严格意义上说cURL并不属于测试工具的范畴。但是由于cURL足够轻量级经常被很多开发人员和测试人员使用所以我在这里做了简单的介绍。
接下来我们再来看看如何使用目前主流的Postman完成API测试。
## 使用图形界面工具Postman进行测试
Postman是目前使用最广泛的Http请求模拟工具之一常常被用于Web Service API的测试。
早期的Postman是以Chrome浏览器的插件plugin形式存在的最新版本的Postman已经是独立的应用了。我猜想是因为这个工具的应用日益广泛所以才有了今天的独立版本。
你可以通过[官方网站](http://www.getpostman.com)下载对应于Mac、Windows和Linux操作系统的不同版本截止文章写作完成时最新的Mac版本是6.2.2。
接下来我就会以Mac 6.2.2版本为例跟你分享如何用Postman完成你的API测试。如果你使用浏览器的plugin版本或者是基于其他操作系统的版本这都没问题基本的操作和步骤都是一样的。
具体的操作,主要包括:
1. 发起API调用
2. 添加结果验证;
3. 保存测试用例;
4. 基于Postman的测试代码自动生成。
**第一步发起API调用**
我们的目标是对Account API做测试所以这里你需要选择Postmant的“Request”模块。进入相应界面后你需要按照图5的提示依次执行以下三步操作发起Account API的调用。
1. 在endpoint输入框中输入“[http://127.0.0.1:8080/account/ID\_008](http://127.0.0.1:8080/account/ID_008)”;
2. 选择“GET”方法
3. 点击“Send”按钮发起API调用。
![](https://static001.geekbang.org/resource/image/d7/b1/d7bc64b03f6296ffabf79ce6c938e1b1.png)
图5 Postman发起Account API的测试
完成以上步骤后界面如图6所示。我们看到返回的response默认以JSON文件的形式显示在下面的Body中。
![](https://static001.geekbang.org/resource/image/db/ce/db132983af623cee5835b01ecd37b5ce.png)
图6 Postman执行GET后的界面
这样就完成了一次Account API的调用是不是非常简单。但问题是这只是一个API调用并没有对调用结果进行自动化验证。接下来我们就加上结果验证的部分一起看看会有什么效果。
**第二步,添加结果验证**
在Postman中添加结果验证也非常方便假定我们在Account API测试过程中有以下四个验证点
1. 请求的返回状态码Status Code应该是200
2. 请求的响应时间应该小于200 ms
3. 请求返回的response header中应该包含“Content-Type”参数
4. 请求返回的response body中“type”的值应该是“friends”
那么接下来我们一起来看看如何使用Postman来添加这四个验证点。
为此我们首先打开“Tests”界面然后在右下角的“SNIPPETS”中依次点击
1. “Status code: Code is 200”
2. “Response time is less than 200 ms”
3. “Response headersContent-Type header check”
4. “Response body: JSON value check”
完成以上操作后“Tests”中会自动生成验证代码接着只要按照具体的测试要求对这些生成的代码进行一些小修改就可以了。
在这个例子中你只需修改需要验证的JSON键值对即可即代码的第15行。修改完成后我们可以再次点击“Send”按钮发起测试。测试通过的界面如图7所示最下面的“Test Results”显示四个测试全部通过。
![](https://static001.geekbang.org/resource/image/69/67/6988a0f511d9742b49f0514f46e1a967.png)
图7 测试通过的界面
**第三步,保存测试用例**
测试通过后我们往往希望可以把这个测试request保存下来以方便后续使用为此Postman提供了保存测试request的功能并提供了Collection来分类管理保存多个测试request。
Collection是用来保存测试request的一个集合Collection内部还可以建立目录结构以方便进一步的分类和管理。
这里我们点击“Save As”按钮在弹出的对话框中可以建立Collection并且可以命名测试request并将其保存到Collection中。
我建立了“API Test Demo”的Collection并且将刚才的测试request命名为“AccountAPI”保存到这个Collection中。
以后再要使用这个测试request时直接在Collection中打开它即可使用。同时你如果申请注册了一个Postman账号就可以很方便地在多个环境中共享这个Collection了。
**第四步基于Postman的测试代码自动生成**
至此你已经掌握了Postman最基本的使用方法但还有一个问题没有解决。很多时候你希望将你的测试request作为回归测试用例集成到CI/CD的流程中这就要求可以通过命令行的方式执行你的测试。为了达到这个目的目前有两种做法
1. **将Postman中的测试request用自动化的方式直接转换成API测试的代码。** 目前Postman已经支持这个功能了可以将保存的测试request自动化转换成常见测试框架直接支持的代码而且支持多语言。
比如基于Java的“OK HTTP”和“Unirest”基于Python的“http.client”和“Requests”基于NodeJS的“Native”“Request”和“Unirest”基于JavaScript的“JQuery AJAX”和“XHR”等等。你可以点击如图8所示的“Code”按钮进入代码生成界面。
![](https://static001.geekbang.org/resource/image/52/41/523124687d3551e1da9bbe300b0faf41.png)
图8 自动生成API测试代码
2. **利用Newman工具直接执行Postman的Collection。** 你需要先将Postman中的Collection导出为JSON文件然后执行以下命令行。
```
newman run examples/sample-collection.json;
```
## 如何应对复杂场景的API测试
我在前面分享的Restful API测试案例中只涉及到了最基本的API的测试方法而且测试场景也很比较简单只是单个API的调用
但在实际项目中除了这种单个API的测试场景外还有很多复杂场景的API测试。所以为了解决你在实际项目中可能会碰到的一些问题我再和你聊聊目前一些常见的典型复杂场景以及相应的测试思路和方法。
**测试场景一被测业务操作是由多个API调用协作完成**
很多情况下一个单一的前端操作可能会触发后端一系列的API调用由于前端测试的相对不稳定性或者由于性能测试的要求你必须直接从后端通过模拟API的顺序调用来模拟测试过程。
这时API的测试用例就不再是简单的单个API调用了而是一系列API的调用并且经常存在后一个API需要使用前一个API返回结果的情况以及需要根据前一个API的返回结果决定后面应该调用哪个API的情况。
好在,**我们已经实现了API的调用和结果解析的代码化这也就意味着我们可以很灵活地直接用代码来处理这些场景了。** 比如通过代码将上个API调用返回的response中的某个值传递给下一个API再比如根据上一个API的返回结果决定下一个应该调用哪个API等。
除此之外,**我们还需要迫切解决的一个问题是如何才能高效地获取单个前端操作所触发的API调用序列。**
解决这个问题的核心思路是通过网络监控的手段捕获单个前端操作所触发的API调用序列。比如通过类似于Fiddler之类的网络抓包工具获取这个调用序列又比如目前很多互联网公司还在考虑基于用户行为日志通过大数据手段来获取这个序列。
**测试场景二API测试过程中的第三方依赖**
API之间是存在依赖关系的比如你的被测对象是API A但是API A的内部调用了API B此时如果由于某种原因API B在被测环境中处于不可用状态那么API A的测试就会受到影响。
在单体架构下通常只会在涉及到第三方API集成的场景中才会遇到这个问题所以还不算严重。但是在微服务架构下API间相互耦合的依赖问题就会非常严重。
解决这个问题的核心思路是启用Mock Server来代替真实的API。那么Mock Server怎么才能真实有效地模拟被替代的API呢这个问题我会在分享《紧跟时代步伐微服务模式下API测试要怎么做》这个主题时和你详细探讨。
**测试场景三异步API的测试**
异步API是指调用后会立即返回但是实际任务并没有真正完成而是需要稍后去查询或者回调Callback的API。
一直以来异步API测试都是API测试中比较困难的部分。**在我看来对异步API的测试主要分为两个部分一是测试异步调用是否成功二是测试异步调用的业务逻辑处理是否正确。**
* 异步调用是否成功,这个还比较简单,主要检查返回值和后台工作线程是否被创建两个方面就可以了。
* 但是对异步调用业务逻辑的测试就比较复杂了因为异步API通常发生在一些比较慢的操作上比如数据库I/O、消息队列I/O等此时测试往往需要去验证数据库中的值、消息队列中的值等这就需要测试代码具有访问和操作数据库或者消息队列的能力。
在实际工程项目中这些能力一般会在测试框架级别提供也就是说要求API测试框架中包含对应的工具类去访问和操作数据库或者消息队列等。
## 总结
通常情况下无论你采用什么API测试工具基本的测试步骤往往都是三步即准备测试数据并不是所有的API测试都需要这一步、通过API测试工具发起对被测API的request、验证返回结果的response。
接下来我通过一个简单的Restful API测试为例和你分享了cURL和Postman这两个常用API测试工具的使用。
其中cURL只具备发起API调用的功能而不具备结果验证能力所以严格地说它并不属于测试工具的范畴。Postman常常被用于Web Service API的测试具体的操作测试流程主要包括发起API调用、添加结果验证、保存测试用例、基于Postman的测试代码自动生成。
最后为了帮你应对实际工程项目中复杂的API测试场景我分享了被测业务操作是由多个API调用协作完成、API测试过程中的第三方依赖、异步API的测试这三个复杂场景下的测试思路和方法。
## 思考题
单个API测试是比较简单的但在实际项目中往往存在按时序的API调用以及异步API调用这类API你是如何测试的遇到过什么难题又是如何解决的
感谢你的收听,欢迎给我留言讨论。