wrk压测
简单介绍
基于系统自带的高性能 I/O 机制,如 epoll, kqueue, 利用异步的事件驱动框架,通过很少的线程就可以压出很大的并发量
wrk 目前仅支持单机压测
基本使用与介绍
WSL:
/mnt/ 目录下 是对应的windows 相应的磁盘
/mnt/e/gitproject/wrk
E:\gitproject\wrk
安装 make 工具
安装 gcc编译环境
sudo apt-get install make
wrk安装
首先在某个目录下 使用 git clone https://github.com/wg/wrk.git 下载
cd wrk目录
然后 进入该目录 使用sudo make
# 将可执行文件移动到 /usr/local/bin 位置
sudo cp wrk /usr/local/bin
验证wrk 是否安装成功
wrk -v
参数介绍
wrk -t12 -c400 -d30s http://127.0.0.1:8080/index.html
简单说一下wrk里面各个参数什么意思?
-t 需要模拟的线程数
-c 需要模拟的连接数
--timeout 超时的时间
-d 测试的持续时间
结果:
Latency:响应时间
Req/Sec:每个线程每秒钟的完成的请求数
Avg:平均
Max:最大
Stdev:标准差
+/- Stdev: 正负一个标准差占比
测试用例
测试环境本机:win10 AMD R7 2700
测试数据:hello world!
测试方法:http get
win10:fastapi+uvicorn
fastapi and uvicorn main:app --reload
wrk -t16 -c100 -d30s http://127.0.0.1:8000
Running 30s test @ http://127.0.0.1:8000
16 threads and 100 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 84.37ms 7.59ms 151.44ms 86.58%
Req/Sec 71.21 15.35 121.00 78.49%
34119 requests in 30.05s, 4.62MB read
Requests/sec: 1135.41
Transfer/sec: 157.46KB
Running 30s test @ http://127.0.0.1:8000
16 threads and 100 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 80.47ms 7.81ms 145.23ms 79.20%
Req/Sec 74.68 17.46 121.00 81.42%
35786 requests in 30.05s, 4.74MB read
Requests/sec: 1191.01
Transfer/sec: 161.67KB
Running 30s test @ http://127.0.0.1:8000
16 threads and 1000 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 861.56ms 57.32ms 1.10s 97.23%
Req/Sec 72.09 39.02 210.00 65.57%
33991 requests in 30.07s, 4.51MB read
Socket errors: connect 0, read 0, write 1963, timeout 0
Requests/sec: 1130.22
Transfer/sec: 153.42KB
win10:doker+fastapi+uvicorn
docker:
Running 30s test @ http://127.0.0.1:8000
16 threads and 100 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 9.28ms 3.95ms 44.69ms 71.35%
Req/Sec 652.48 80.30 0.91k 68.48%
312592 requests in 30.10s, 61.11MB read
Requests/sec: 10385.30
Transfer/sec: 2.03MB
Running 30s test @ http://127.0.0.1:8000
12 threads and 200 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 17.27ms 7.15ms 66.18ms 70.56%
Req/Sec 0.93k 87.26 1.26k 68.50%
334625 requests in 30.10s, 65.42MB read
Requests/sec: 11117.33
Transfer/sec: 2.17MB
win10:go http
win10:
wrk -t16 -c1000 -d30s http://127.0.0.1:8001
Running 30s test @ http://127.0.0.1:8001
16 threads and 100 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 1.10ms 705.16us 14.69ms 87.90%
Req/Sec 5.72k 363.31 9.16k 76.72%
2737464 requests in 30.10s, 336.77MB read
Requests/sec: 90946.16
Transfer/sec: 11.19MB
Running 30s test @ http://127.0.0.1:8001
10 threads and 100 connections
Thread Stats Avg Stdev Max +/- Stdev
Latency 1.15ms 745.51us 14.42ms 87.68%
Req/Sec 8.87k 755.39 24.24k 80.25%
2649469 requests in 30.10s, 325.95MB read
Requests/sec: 88018.88
Transfer/sec: 10.83MB
使用lua脚本进行复杂测试
使用post 请求并带有参数 可以使用带有lua的脚本来指定 通过参数--script
指定lua脚本
wrk 支持在三个阶段对压测进行个性化,分别是启动阶段、运行阶段和结束阶段。每个测试线程,都拥有独立的Lua 运行环境
启动阶段
function setup(thread)
在脚本文件中实现 setup 方法,wrk 就会在测试线程已经初始化,但还没有启动的时候调用该方法。
wrk会为每一个测试线程调用一次 setup 方法,并传入代表测试线程的对象 thread 作为参数。
setup 方法中可操作该 thread 对象,获取信息、存储信息、甚至关闭该线程
thread.addr - get or set the thread's server address
thread:get(name) - get the value of a global in the thread's env
thread:set(name, value) - set the value of a global in the thread's env
thread:stop() - stop the thread
运行阶段
function init(args)
function delay()
function request()
function response(status, headers, body)
init(args)
: 由测试线程调用,只会在进入运行阶段时,调用一次。支持从启动 wrk 的命令中,获取命令行参数;delay()
: 在每次发送请求之前调用,如果需要定制延迟时间,可以在这个方法中设置;request()
: 用来生成请求, 每一次请求都会调用该方法,所以注意不要在该方法中做耗时的操作;response(status, headers, body)
: 在每次收到一个响应时被调用,为提升性能,如果没有定义该方法,那么wrk
不会解析headers
和body
;结束阶段
function done(summary, latency, requests)
done()
方法在整个测试过程中只会被调用一次,我们可以从给定的参数中,获取压测结果,生成定制化的测试报告。
自定义 Lua 脚本中可访问的变量以及方法
wrk
变量
wrk = {
scheme = "http",
host = "localhost",
port = 8080,
method = "GET",
path = "/",
headers = {},
body = nil,
thread = <userdata>,
}
以上定义了一个 table 类型的全局变量,修改该 wrk 变量,会影响所有请求。
方法:
wrk.fomat
wrk.lookup
wrk.connect
function wrk.format(method, path, headers, body)
wrk.format returns a HTTP request string containing the passed parameters
merged with values from the wrk table.
# 根据参数和全局变量 wrk,生成一个 HTTP rquest 字符串。
function wrk.lookup(host, service)
wrk.lookup returns a table containing all known addresses for the host
and service pair. This corresponds to the POSIX getaddrinfo() function.
# 给定 host 和 service(port/well known service name),返回所有可用的服务器地址信息。
function wrk.connect(addr)
wrk.connect returns true if the address can be connected to, otherwise
it returns false. The address must be one returned from wrk.lookup().
# 测试给定的服务器地址信息是否可以成功创建连接
通过 Lua 脚本压测示例
调用 POST 接口:
wrk.method = "POST"
wrk.body = "foo=bar&baz=quux"
wrk.headers["Content-Type"] = "application/x-www-form-urlencoded"
注意: wrk 是个全局变量,这里对其做了修改,使得所有请求都使用 POST 的方式,并指定了 body 和 Content-Type头
自定义每次请求的参数:
request = function()
uid = math.random(1, 10000000)
path = "/test?uid=" .. uid
return wrk.format(nil, path)
end
在 request 方法中,随机生成 1~10000000 之间的 uid,并动态生成请求 URL.
每次请求前,延迟 10ms:
function delay()
return 10
end
请求的接口需要先进行认证,获取 token 后,才能发起请求,咋办?
token = nil
path = "/auth"
request = function()
return wrk.format("GET", path)
end
response = function(status, headers, body)
if not token and status == 200 then
token = headers["X-Token"]
path = "/test"
wrk.headers["X-Token"] = token
end
end
上面的脚本表示,在 token 为空的情况下,先请求 /auth
接口来认证,获取 token, 拿到 token 以后,将 token 放置到请求头中,再请求真正需要压测的/test
接口。
压测支持 HTTP pipeline 的服务: ```lua init = function(args) local r = {} r[1] = wrk.format(nil, “/?foo”) r[2] = wrk.format(nil, “/?bar”) r[3] = wrk.format(nil, “/?baz”)
req = table.concat(r) end
request = function() return req end ``` 通过在 init 方法中将三个 HTTP请求拼接在一起,实现每次发送三个请求,以使用 HTTP pipeline。