Gatling 是基于 AKKA 和 Netty 开发的一款高性能压测工具,本章节主要讲述 Gatling 提供的 API 和一些基本概念
gatling 主要压测的关键在于场景设置,这决定了你要干什么和你的目标
一般可以使用以下两种方法配置断言和协议:
assertions
:在模拟中设置断言protocols
:设置协议定义在使用 gatling 的时候我们会发现有时候 gatling 的一些注入方案和我们设想的方便并不相同,使得我们的方案出现偏差或者交流问题。前面说过我曾经加入过一个 Gatling 的技术分享群,我发现其实很多人并没有理解 gatling 中的开放模型 (Open Model) 和封闭模型 (Closed Model) 是什么意思,所以这里会对 gatling 的两种模型进行描述,我的理解是这两种模型分别代表了现在主流的两种压测模式 RPS 和并发这两种模式,关于这两种模式的差异我在这里面不做过多的描述,大家可以看这篇文章,里面有具体的描述传送门
开放模型 (Open Model)
开放模型中我们可以主动的控制用户的到达率,这个和我们 jmeter 或者 ab 的并发概念会有出入,次模型中涉及的方法均为 RPS 的压测模式,使用开放模型会有什么好处呢?
开放模型的坏处:
封闭模型
封闭模型可控制并发用户数,这与 jmeter 的线程组和 ab 的并发概念相同,保持一定的连接数并复用,减少因为建立连接所导致的问题,满负荷运行时,新用户只有在另一个用户退出后才能有效地进入系统。
封闭模型的好处:
封闭模型的坏处:
Gatling 是通过 inject 注入方式来模拟用户操作的场景
开放模型
setUp(
scn.inject(
nothingFor(4 seconds), // 1
atOnceUsers(10), // 2
rampUsers(10) during (5 seconds), // 3
constantUsersPerSec(20) during (15 seconds), // 4
constantUsersPerSec(20) during (15 seconds) randomized, // 5
rampUsersPerSec(10) to 20 during (10 minutes), // 6
rampUsersPerSec(10) to 20 during (10 minutes) randomized, // 7
heavisideUsers(1000) during (20 seconds) // 8
).protocols(httpProtocol)
)
nothingFor(duration)
:设置一段停止的时间,这段时间什么都不做atOnceUsers(nbUsers)
:立即注入一定数量的虚拟用户rampUsers(nbUsers) during(duration)
:在指定时间内,设置一定数量逐步注入的虚拟用户constantUsersPerSec(rate) during(duration)
:定义一个在每秒钟恒定的并发用户数,持续指定的时间constantUsersPerSec(rate) during(duration) randomized
:定义一个在每秒钟围绕指定并发数随机增减的并发,持续指定时间rampUsersPerSec(rate1) to (rate2) during(duration)
:定义一个并发数区间,运行指定时间,并发增长的周期是一个规律的值rampUsersPerSec(rate1) to(rate2) during(duration) randomized
:定义一个并发数区间,运行指定时间,并发增长的周期是一个随机的值 heavisideUsers(nbUsers) during(duration)
:按照延伸到给定持续时间的重阶梯函数的平滑近似值,注入给定数量的用户封闭模式
setUp(
scn.inject(
constantConcurrentUsers(10) during (10 seconds), // 1
rampConcurrentUsers(10) to (20) during (10 seconds) // 2
)
)
constantConcurrentUsers(nbUsers) during(duration)
: 在指定时间段内保持恒定的并发用户数量rampConcurrentUsers(10) to (20) during (10 seconds)
: 注入,以便系统中的并发用户数从一个数字线性增加到另一个 ==注意点==
开放模型和封闭模型在工作负载模型具有相反的含义,不能将它们混合在同一 Injection 配置文件中
Meta DSL
在我们日常的工作中有时候并不知道接口的性能瓶颈在哪里,当我们使用开放模型测试时,会不停的进行重复的操作来定义每次爬升的值,为了解决这个问题 Gatling 在 3.0 开始添加了一种Meta DSL
新的方法来方便我们操作
java
// generate an open workload injection profile
// with levels of 10, 15, 20, 25 and 30 arriving users per second
// each level lasting 10 seconds
// separated by linear ramps lasting 10 seconds
setUp(
scn.inject(
incrementUsersPerSec(5) // Double
.times(5)
.eachLevelLasting(10 seconds)
.separatedByRampsLasting(10 seconds)
.startingFrom(10) // Double
)
)
java
// generate a closed workload injection profile
// with levels of 10, 15, 20, 25 and 30 concurrent users
// each level lasting 10 seconds
// separated by linear ramps lasting 10 seconds
setUp(
scn.inject(
incrementConcurrentUsers(5) // Int
.times(5)
.eachLevelLasting(10 seconds)
.separatedByRampsLasting(10 seconds)
.startingFrom(10) // Int
)
)
incrementUsersPerSec 用于开放模型负载测试和 incrementConcurrentUsers 封闭模型负载 (users/sec vs concurrent users)
separatedByRampsLasting 并且 startingFrom 都是可选的。如果未指定坡度,则测试将在完成后立即从一个级别跳到另一级别。如果未指定起始用户数,则测试将从 0 个并发用户或 0 个用户/秒开始,并立即进行下一步。
当我们需要使用暂停时,可以使用Simulation
的多种方法配置暂停,而且暂停定义也可以放在 scenario 中进行配置:
disablePauses
:禁用模拟暂停constantPauses
:每个暂停的持续时间正是 pause(duration) 元素中指定的持续时间。exponentialPauses
:每个暂停的持续时间平均为 pause(duration) 元素中指定的持续时间,并遵循指数分布。normalPausesWithStdDevDuration(stdDev: Duration)
:每个暂停的持续时间平均为 pause(duration) 元素中指定的时间,并遵循正态分布。stdDev 是持续时间。normalPausesWithPercentageDuration(stdDev: Double)
:每个暂停的持续时间平均为 pause(duration) 元素中指定的时间,并遵循正态分布。stdDev 是暂停值的百分比。customPauses(custom: Expression[Long])
:暂停持续时间由提供的计算 Expression[Long]。在这种情况下,将跳过填充时间。uniformPausesPlusOrMinusPercentage(plusOrMinus: Double)
和 uniformPausesPlusOrMinusDuration(plusOrMinus: Duration)
:每个暂停的持续时间平均为元素中指定的时间,并且遵循均匀分布setUp(
...
).protocols(protocol)
.pauses(uniformPausesPlusOrMinusPercentage(1))
.disablePauses
.constantPauses
.exponentialPauses
.uniformPauses(1.5)
.uniformPauses(1337 seconds)
如果需要 RPS 而不是并发用户数进行压测,可以使用 constantUsersPerSec(…) 来设置用户的到达率,因此无需节流即可设置请求,因为这在大多数情况下都是多余的,如果由于某种原因这还不够,那么可以使用 Gatling 的 throttle 方法进行节流,节流是按协议实现的,并支持常规 HTTP 和 JMS 请求
==注意==
仍然必须在 scenario 注入用户。节流尝试使用给定 scenario 及其注入配置文件(用户数和持续时间)来确保目标吞吐量。这是一个上限,如果当预设值没有达到指定的,将无法达到节流所设置的目标。如果施压时间少于节流设置的时间,则在所有用户完成后施压会主动停止,如果的设置的持续时间超过节流时间,则施压将在节流所设置的末端停止
也可以根据情况配置节流。
setUp(scn.inject(constantUsersPerSec(100) during (30 minutes))).throttle(
reachRps(100) in (10 seconds),
holdFor(1 minute),
jumpToRps(50),
holdFor(2 hours)
)
此段代码是模拟为 10 秒内达到 100req/s,然后保持 1 分钟这个吞吐量后请求量跳到 50 req/s 并保持两小时
节流的基础是:
reachRps(target) in (duration)
:在指定的 RPS 达到指定目标
jumpToRps(target)
:立即跳转到指定的目标 RPS
holdFor(duration)
:在指定的时间段内保持当前的吞吐量
在有时候压测过程中,会因为各种原因我们无法估计脚本结束的时间,例如当服务端无法处理请求且未设置
maxDuration 即使某些虚拟用户仍在运行,您也可以根据持续时间限制来强制终止运行。
如果您需要在无法预测的情况下限制仿真的持续时间,这将非常有用。
setUp(scn.inject(rampUsers(1000) during (20 minutes))).maxDuration(10 minutes)