強(qiáng)大的負(fù)載測(cè)試工具之Gatling
在應(yīng)用程序上線之前,有多少人做過性能測(cè)試?
估計(jì)大部分開發(fā)者更多地關(guān)注功能測(cè)試,并且會(huì)提供一些單元測(cè)試和集成測(cè)試的用例。然而,有時(shí)候性能漏洞導(dǎo)致的影響比未發(fā)現(xiàn)的業(yè)務(wù)漏洞更嚴(yán)重,因?yàn)樾阅苈┒从绊懙氖钦麄€(gè)系統(tǒng),而不僅僅是一個(gè)業(yè)務(wù)進(jìn)程。
可能你們很多人聽過 JMeter ,但是今天將介紹有競(jìng)爭(zhēng)力的解決方案 —— Gatling 。它能生成豐富多彩的報(bào)告,包含測(cè)試案例中收集的所有指標(biāo)。該功能似乎比 JMeter 更好。
在討論 Gatling 之前,先了解下理論知識(shí),性能測(cè)試的兩種類型,負(fù)載測(cè)試和壓力測(cè)試:
負(fù)載測(cè)試(Load Testing):負(fù)載測(cè)試是一種主要為了測(cè)試軟件系統(tǒng)是否達(dá)到需求文檔設(shè)計(jì)的目標(biāo),譬如軟件在一定時(shí)期內(nèi),***支持多少并發(fā)用戶數(shù),軟件請(qǐng)求出錯(cuò)率等,測(cè)試的主要是軟件系統(tǒng)的性能。
壓力測(cè)試(Stress Testing):壓力測(cè)試主要是為了測(cè)試硬件系統(tǒng)是否達(dá)到需求文檔設(shè)計(jì)的性能目標(biāo),譬如在一定時(shí)期內(nèi),系統(tǒng)的cpu利用率,內(nèi)存使用率,磁盤I/O吞吐率,網(wǎng)絡(luò)吞吐量等,壓力測(cè)試和負(fù)載測(cè)試***的差別在于測(cè)試目的不同。
Gatling 簡(jiǎn)介
Gatling 是一個(gè)功能強(qiáng)大的負(fù)載測(cè)試工具。它是為易用性、可維護(hù)性和高性能而設(shè)計(jì)的。
開箱即用,Gatling 帶有對(duì) HTTP 協(xié)議的出色支持,使其成為負(fù)載測(cè)試任何 HTTP 服務(wù)器的***工具。由于核心引擎實(shí)際上是協(xié)議不可知的,所以完全可以實(shí)現(xiàn)對(duì)其他協(xié)議的支持,例如,Gatling 目前也提供JMS 支持。
只要底層協(xié)議(如 HTTP)能夠以非阻塞的方式實(shí)現(xiàn),Gatling 的架構(gòu)就是異步的。這種架構(gòu)可以將虛擬用戶作為消息而不是專用線程來實(shí)現(xiàn)。因此,運(yùn)行數(shù)千個(gè)并發(fā)的虛擬用戶不是問題。
Gatling 快速入門實(shí)踐
1、創(chuàng)建 Spring Boot 應(yīng)用,提供 RESTful API,以供測(cè)試
https://github.com/ChinaSilence/gatling-test.git
如果有自己測(cè)試的 Web 應(yīng)用可以忽略本步驟!
2、啟動(dòng)數(shù)據(jù)庫
Github 中的示例代碼依賴了 PostgresSQL,所以要先啟動(dòng)數(shù)據(jù)庫,最簡(jiǎn)單的方式當(dāng)然是用 Docker 咯:
- docker run -d \
- --name postgres \
- -e POSTGRES_DB=gatling \
- -e POSTGRES_USER=gatling \
- -e POSTGRES_PASSWORD=gatling123 \
- -p 5432:5432 \
- postgres
3、在 IDEA 中安裝 scala 環(huán)境
安裝 scala 插件
安裝 scala SDK
4、編寫性能測(cè)試腳本
每一個(gè) Gatling 測(cè)試都要繼承 Simulation 類,在里面你可以使用Gatling Scala DSL 來聲明一個(gè)場(chǎng)景列表。這里的目標(biāo)是運(yùn)行 30 個(gè)客戶端,同時(shí)發(fā)送 1000 次請(qǐng)求。首先,客戶端通過調(diào)用 POST /persons 方法將添加數(shù)據(jù)到數(shù)據(jù)庫中;然后,嘗試通過調(diào)用 GET /persons/{id} 方法使用 id 來查詢數(shù)據(jù)。
- class ApiGatlingSimulationTest extends Simulation {
- val scn = scenario("AddAndFindPersons").repeat(1000, "n") {
- exec(
- http("AddPerson-API")
- .post("http://localhost:8080/persons")
- .header("Content-Type", "application/json")
- .body(StringBody("""{"firstName":"John${n}","lastName":"Smith${n}","birthDate":"1980-01-01", "address": {"country":"pl","city":"Warsaw","street":"Test${n}","postalCode":"02-200","houseNo":${n}}}"""))
- .check(status.is(200))
- ).pause(Duration.apply(5, TimeUnit.MILLISECONDS))
- }.repeat(1000, "n") {
- exec(
- http("GetPerson-API")
- .get("http://localhost:8080/persons/${n}")
- .check(status.is(200))
- )
- }
- setUp(scn.inject(atOnceUsers(30))).maxDuration(FiniteDuration.apply(10, "minutes"))
- }
5、運(yùn)行 Spring Boot 應(yīng)用
6、運(yùn)行測(cè)試腳本
配置 Maven 插件參數(shù)
- <build>
- <plugins>
- <plugin>
- <groupId>io.gatling</groupId>
- <artifactId>gatling-maven-plugin</artifactId>
- <version>${gatling-plugin.version}</version>
- <configuration>
- <!-- 測(cè)試腳本 -->
- <simulationClass>com.anoyi.test.ApiGatlingSimulationTest</simulationClass>
- <!-- 結(jié)果輸出地址 -->
- <resultsFolder>/Users/admin/code/gatling</resultsFolder>
- </configuration>
- </plugin>
- </plugins>
- </build>
執(zhí)行測(cè)試
- mvn gatling:execute
7、查看測(cè)試報(bào)告
全局報(bào)告
單個(gè)接口明細(xì)報(bào)告
【本文為51CTO專欄作者“翟永超”的原創(chuàng)稿件,轉(zhuǎn)載請(qǐng)通過51CTO聯(lián)系作者獲取授權(quán)】