Gatling压测案例

引言:公司某个网站需要做压测,验证视频分片功能承载能力,刚开始选择了Apache Jmeter,发现Jmeter占用资源太大了,在虚拟机跑500个线程并发就内存溢出了。因为Jmeter一条线程就是一条用户线程,每条线程都需要占用一定空间的内存,因此在高并发量下需要大量资源。一个资深同事建议试试Gatling,它是异步非阻塞线程,对资源占用小且速度个人感觉比Jmeter快,缺点是没有GUI,需要自己写脚本

Windows测试

直接在官网下一个gatling-maven-plugin-demo-java-main.zip,里面有一个可以直接运行的demo

java 复制代码
package example;

import static io.gatling.javaapi.core.CoreDsl.*;
import static io.gatling.javaapi.http.HttpDsl.*;

import io.gatling.javaapi.core.*;
import io.gatling.javaapi.http.*;

public class BasicSimulation extends Simulation {

  // Load VU count from system properties
  // Reference: https://docs.gatling.io/guides/passing-parameters/
  private static final int vu = Integer.getInteger("vu", 1);

  // Define HTTP configuration
  // Reference: https://docs.gatling.io/reference/script/protocols/http/protocol/
  private static final HttpProtocolBuilder httpProtocol = http.baseUrl("https://api-ecomm.gatling.io")
      .acceptHeader("application/json")
      .userAgentHeader(
          "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/134.0.0.0 Safari/537.36");

  // Define scenario
  // Reference: https://docs.gatling.io/reference/script/core/scenario/
  private static final ScenarioBuilder scenario = scenario("Scenario").exec(http("Session").get("/session"));

  // Define assertions
  // Reference: https://docs.gatling.io/reference/script/core/assertions/
  private static final Assertion assertion = global().failedRequests().count().lt(1L);

  // Define injection profile and execute the test
  // Reference: https://docs.gatling.io/reference/script/core/injection/
  {
    setUp(scenario.injectOpen(atOnceUsers(vu))).assertions(assertion).protocols(httpProtocol);
  }
}

直接在terminal执行命令./mvnw clean install ./mvnw gatling:test即可

如果是在cmd里面需要更改一下命令mvnw.cmd gatling:test

Demo里面定义了默认并发数量是1
private static final int vu = Integer.getInteger("vu", 1);

可以直接改成自己想要的并发数,也可以通过options指定

如500并发数:./mvnw gatling:test -Dvu=500

Linux测试

在Linux测试最方便就是用docker跑。

1.先准备好.scale测试脚本

(上面的demo是java写的,现在需要一个scale写的脚本)

例如我自己写的关于视频切片功能测试的脚本:

scale 复制代码
package example

import io.gatling.core.Predef._
import io.gatling.http.Predef._
import scala.concurrent.duration.Duration.Inf
import scala.util.Random

class BasicSimulation extends Simulation {

  // 视频大小 (例如 800MB) ,不要超过目标视频实际大小
  val VIDEO_SIZE_BYTES = 800L * 1024 * 1024

  // 模拟用户每次拖动加载的数据块大小 (例如 3MB)
  val CHUNK_SIZE_BYTES = 3L * 1024 * 1024

  // 获取虚拟用户数量,优先读取系统属性 -Dvu=xx,否则默认为 1
  val vu = Option(System.getProperty("vu")).map(_.toInt).getOrElse(1)

  // 1. 定义 HTTP 协议配置
  val httpProtocol = http
    .baseUrl("http://10.211.2.123:8080")
    .userAgentHeader("Mozilla/5.0 (Windows NT 10.0; Win64; x64) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/123.0.0.0 Safari/537.36")
    .acceptHeader("*/*")
    .connectionHeader("keep-alive")

  // 2. 定义场景
  val myScenario = scenario("Video Range Request")
    .exec { session =>
      // 生成 0 到 (视频大小 - 块大小) 之间的随机数
      val start = Math.abs(Random.nextLong()) % (VIDEO_SIZE_BYTES - CHUNK_SIZE_BYTES)
      val end = start + CHUNK_SIZE_BYTES - 1
      val rangeValue = s"bytes=$start-$end"
      println(s"Range: $rangeValue")
      session.set("rangeHeader", rangeValue)
    }
    .exec(
      http("Get Random Video Chunk")
        // 视频path
        .get("/test.mp4")
        .header("Range", "${rangeHeader}")
        .header("Referer", "http://10.211.2.123:8080/test.mp4")
        .check(
          bodyBytes.transform { (bytes: Array[Byte]) =>
            println(s"Received ${bytes.length} bytes")
            bytes
          }
        )
        .check(status.in(200, 206))
    )

  // 3. assertion
  val assertion = global.failedRequests.count.lt(1)

  // 4. start
  setUp(
    myScenario.inject(
      atOnceUsers(vu)
    )
  ).protocols(httpProtocol)
   .assertions(assertion)
}

2.准备好文件目录

在 Linux 服务器上,创建一个文件夹(比如叫 my-test),并按照 Gatling 的标准结构放好文件:

复制代码
my-test/
├── results/                # (空文件夹,用来放报告)
└── user-files/             # 关键目录
    └── simulations/        # 把你的scala script放这里
        └── BasicSimulation.scala

3.开启容器测试

进入 my-test 目录,run command

bash 复制代码
docker run -it --rm \
  -e JAVA_OPTS="-Dvu=10" \
  -v $(pwd)/results:/opt/gatling/results \
  -v $(pwd)/user-files:/opt/gatling/user-files \
  denvazh/gatling
  • -v $(pwd)/user-files:/opt/gatling/user-files

    这句话的意思是:"把服务器上当前目录下的 user-files 文件夹,替换掉容器里原本的 user-files 文件夹"。

  • 这样,容器启动后,就能直接看到你上传的 .scala 脚本了。

  • JAVA_OPTS="-Dvu=10" 指定concurrent数量

如出现Select run description (optional),enter即可继续

相关推荐
Zella折耳根17 小时前
复习篇-常用实用类
java
devilnumber1 天前
Java 递归算法 详解 + 核心要点 + 实战运用 + 避坑指南
java·开发语言·算法
asdfg12589631 天前
JavaBean是什么?怎么理解?有什么用途?
java·开发语言
摇滚侠1 天前
SpringMVC 入门到实战 文件上传 75-77
java·后端·spring·maven·intellij-idea
GIS数据转换器1 天前
城市排水生命线安全运行监测平台深度解析
java·运维·人工智能·python·安全·数据挖掘·无人机
华如锦1 天前
面了很多 Java转AI Agent方向,一些面试题总结
java·开发语言·人工智能·python·ai
睡不醒男孩0308231 天前
CLup 6.x 版本中针对StarRocks 存算一体集群的完整操作手册
java·服务器·网络·clup
程序员黑豆1 天前
Java中怎么实现字符串拼接呢【AI全栈开发】
java
java1234_小锋1 天前
LangChain4j 开发Java Agent智能体- 多模态支持
java·开发语言·langchain4j