项目背景

最近有一个超级表格的项目测试,支持多人实时协作编辑表格。项目选用了websocket这种基于 tcp 的全双工通讯协议,它可以实现服务端和客户端之间的主动数据推送,具有资源占用少、实时性高等特点,消息推送模式包括:单对单、单对多(全体广播)。

我们的项目使用websocket协议提交、广播增量数据,并广播提示其他用户拉取全量数据以达到同步的目的。

在性能测试中需要模拟多个用户协同编辑一个文档的场景,选用jmeter作为测试工具。

jmeter插件

在jmeter中测试websocket协议接口,需要先安装websocket插件和依赖包;

需添加的插件为:
jmeter-websocket-samplers-1.2.1.jar

依赖包:jetty-http

jetty-io

jetty-util

websocket-api

websocket-client

websocket-common

把插件放到jmeter的libext,依赖包放到jmeter的lib目录,打开jmeter即可看到websocket相关的取样器。

这几种取样器的作用如下表所示:

常用的取样器是websocket open connection和websocket request-response sampler,前者用来建立连接不发送数据,后者建立连接并发送data或者只发data。

创建脚本

在编写脚本前先通过浏览器查看建立连接的过程,首先打开一个文档时通过http升级机制,使用http的upgrade和connection协议头的方式将连接升级为websocket。

返回的状态码为101则说明websocket连接建立成功,协议已经变为了websocket,如下图所示,浏览器先后发送三个数据帧,包括connect建立连接、subscribe订阅文档、send类型为client_ready消息,客户端收到的数据包括服务端返回的响应和每25s一个的心跳帧。

后续通过websocket协议提交、广播、拉取各种类型的增量数据。

了解了这些之后就可以开始在测试场景中添加和配置websocket的sampler编写脚本。

1.右键单击测试计划,选择添加->线程->线程组,将一个线程组添加到测试计划。

2.在线程组添加websocket request-response sampler建立连接,选择建立新连接,并发送请求数据如下:

说明:

(1)connection:有两个选项,第一项是使用已有连接,就是上一个websocket请求所建立的连接通道,选择后server url全置灰只读不可操作。第二项是新建连接通道。

(2)server url:可以发送ws协议和加密的websocket,即wss协议。server name or ip处填写websocket服务器的地址,port和path依照实际填写即可。

(3)request data:支持文本(包括json)和binary二进制数据的发送,可以将浏览器中查看到的数据根据需求参数化之后填写。默认请求响应的超时时间为6s,超过这个时间报错。

因为要通过收到返回包含connected字样的报文来判断是否发送后续的消息。需要把收到的服务端返回的其他数据帧过滤掉,再用断言进行判断。

右键单击websocket request-response sampler添加->配置元件->websocket text frame filter,配置条件过滤掉不需要的数据帧,如心跳帧等。

右键添加->断言->响应断言,添加断言对建立连接的响应帧进行判断。

第一个建立连接的请求成功后,后续websocket request-response sampler的连接类型选择使用已存在的连接,下面的server url全置灰只读不可操作。

添加后续的subscribe订阅文档,通过固定请求数据中的文档id可以使所有用户打开同一个文档,添加send类型为client_ready消息请求后,就完成了基本的打开文档建立连接的操作,之后可以添加循环控制器并用类似的方法添加发送编辑文档的请求。

运行脚本

脚本完成之后配置线程组的线程数,循环次数或执行时间,即可模拟并发用户同时编辑文档,添加结果树、聚合报告等监听器后可以查看测试的运行情况:

从浏览器也可以验证收到了其他并发用户广播的数据。

除了多用户协同编辑用一个文档的场景,性能测试的过程中还覆盖了协同用户保存同一文档、不同文档同时保存等场景,确保测试场景的充分性。