多轮对话压测#
多轮对话压测功能允许用户在真实的多轮交互场景下测试模型服务。与普通压测不同,多轮模式会将模型的真实回复追加到上下文,后续每轮请求都携带完整的历史对话,从而真实模拟用户与模型的连续对话过程,并能评估服务在上下文不断增长时的延迟、吞吐量及 KV 缓存命中率表现。
功能特性#
真实上下文累积:每轮请求成功后,将模型实际输出追加到对话历史,下一轮请求携带完整历史,而非仅发送当前用户消息。
KV 缓存可命中率估算:基于客户端 token 计数,估算每轮请求中历史对话 token 占总输入 token 的比例,即理论上可被服务端 prefix caching 利用的上限;实际是否命中取决于服务端是否开启并维持了对应缓存。
多数据集支持:提供随机合成(
random_multi_turn)、真实对话(share_gpt_zh_multi_turn/share_gpt_en_multi_turn)和自定义本地数据(custom_multi_turn)四种数据集。参数语义一致:
--number为总 turn 数,--parallel为并发 turn 数,与普通压测模式语义保持一致。
参数说明#
多轮专属参数#
参数 |
类型 |
说明 |
默认值 |
|---|---|---|---|
|
|
启用多轮对话压测模式 |
|
|
|
每个对话最少用户轮数,仅 |
|
|
|
每个对话最多用户轮数; |
|
已有参数在多轮模式下的语义#
参数 |
多轮模式下的含义 |
|---|---|
|
总发送 turn 数(而非对话数);所有 worker 合计发送的 HTTP 请求数达到此值后停止 |
|
同时在途的 turn 级并发请求数 |
数据集说明#
random_multi_turn#
基于 random 数据集随机生成 token 序列,每个对话包含 [min_turns, max_turns] 轮用户消息,无需外部数据文件,适合快速基准测试和性能对比。
必需参数:--tokenizer-path、--max-turns
可选参数:--min-turns(默认 1)、--min-prompt-length、--max-prompt-length(控制每轮用户消息的 token 长度范围)
数据集产出的每条对话结构如下:
[
{"role": "user", "content": "...turn 1 随机 token 序列..."},
{"role": "user", "content": "...turn 2 随机 token 序列..."}
]
注意:
--tokenize-prompt在多轮模式下不支持,会被自动忽略。多轮对话始终通过/v1/chat/completions端点以消息列表形式发送。
custom_multi_turn#
使用本地 JSONL 文件作为自定义多轮对话数据集,每行直接以 OpenAI messages 格式存储一条完整对话,无需任何格式转换,适合已有对话数据直接压测的场景。
必须指定
--dataset-path指向本地 JSONL 文件可选截断:通过
--max-turns限制每条对话最多使用的用户轮数
JSONL 数据集格式(每行一条对话,为 OpenAI messages 数组):
[{"role": "user", "content": "你好"}, {"role": "assistant", "content": "你好!有什么可以帮助你?"}, {"role": "user", "content": "帮我写一首诗"}]
[{"role": "user", "content": "What is the capital of France?"}, {"role": "assistant", "content": "Paris."}, {"role": "user", "content": "Tell me more about it."}]
每行需满足:
必须是一个 JSON 数组
每个元素必须包含
role和content字段role取值为user或assistant至少包含一个
user消息
运行时上下文结构(第 2 轮发送时):
[
{"role": "user", "content": "你好"},
{"role": "assistant", "content": "<模型第 1 轮实际回复>"},
{"role": "user", "content": "帮我写一首诗"}
]
说明:数据集中的
assistant消息仅用于标识对话结构,不会被直接发送给模型。运行时 worker 始终将模型的实际输出追加到上下文,保证历史准确。
工作流程#
加载对话池:启动时从数据集文件顺序读取,将所有对话预加载到内存,最多加载
--number条(防止大数据集占用过多内存)。启动 workers:创建
--parallel个并发协程(worker),各自独立运行,所有 worker 共享同一个全局 turn 计数器。对话分配(先到先得,顺序轮询):对话按顺序分配给 worker,谁先完成当前对话,谁先取下一条。所有对话用完后从头循环复用。
以
--parallel 3、共 5 条对话为例:启动时:3 个 worker 同时开始 Worker 0 → 取第 1 条对话 Worker 1 → 取第 2 条对话 Worker 2 → 取第 3 条对话 ↓ 各自发送请求(等待回复期间其他 worker 可以继续执行) 假设 Worker 1 最先跑完第 2 条对话的所有轮次: Worker 1 → 取第 4 条对话 假设 Worker 0 第二个跑完第 1 条对话: Worker 0 → 取第 5 条对话 第 5 条用完后若还未达到 --number: Worker 0 → 循环回到第 1 条对话重新开始 ... 以此类推,谁先跑完当前对话,谁先取下一条单条对话执行(逐 turn):每个 worker 持有自己独立的对话历史,不与其他 worker 共享:
第 1 轮:发送 [用户消息1] → 收到模型回复1 第 2 轮:发送 [用户消息1, 模型回复1, 用户消息2] → 收到模型回复2 第 3 轮:发送 [用户消息1, 模型回复1, 用户消息2, 模型回复2, 用户消息3] → ...每轮将用户消息追加到历史后发送请求,收到模型的真实回复后追加到历史,供下一轮携带。数据集中原有的 assistant 内容不会发给模型,仅用模型实际输出构建上下文。
预算控制与停止:全局 turn 计数在发送请求前同步递增,确保所有 worker 合计发送的请求数不超过
--number。一旦达到上限,所有 worker 停止,不再取新对话。失败处理:某轮请求失败时,当前对话立即放弃,worker 取下一条新对话重新开始,不会将失败的上下文带入后续请求。
指标汇总:所有 worker 完成后,汇总全部延迟、吞吐量及多轮专属指标(平均上下文轮数、KV 缓存命中率)后输出结果。
注意:请求成功率低于 100% 时,被中断的对话不会贡献后续轮次的上下文,KV 缓存命中率等指标可能偏低。
输出指标说明#
多轮模式在普通压测的全部指标基础上,额外输出以下两个专属指标:
指标 |
说明 |
如何解读 |
|---|---|---|
|
每次请求时,上下文中平均包含的用户轮数 |
反映测试过程中上下文平均增长程度;值越大说明对话越深入,prompt 越长 |
|
估算当前请求中历史对话 token 占总输入 token 的比例(即理论上可被 prefix caching 利用的上限) |
比值越高说明历史上下文占比越大,若服务端开启了 prefix caching 则收益越显著;计算方式: |
其余指标(Avg/P99 Latency、TTFT、TPOT、RPS、TPS)含义与普通压测一致。
使用示例#
1. 使用 random_multi_turn(合成多轮对话)#
适用场景:快速评测服务在指定 prompt 长度分布和多轮深度下的性能,无需真实数据集。
evalscope perf \
--model Qwen2.5-7B-Instruct \
--tokenizer-path Qwen/Qwen2.5-7B-Instruct \
--url http://127.0.0.1:8801/v1/chat/completions \
--api openai \
--dataset random_multi_turn \
--min-prompt-length 256 \
--max-prompt-length 512 \
--max-tokens 256 \
--multi-turn \
--min-turns 2 \
--max-turns 5 \
--number 20 \
--parallel 10
输出示例:
Detailed Performance Metrics
┏━━━━━━┳━━━━━━┳━━━━━━┳━━━━━━━━━┳━━━━━━━━━┳━━━━━━━━━┳━━━━━━━━━┳━━━━━━━━━┳━━━━━━━━┳━━━━━━━━━┳━━━━━━━━┓
┃ ┃ ┃ ┃ Avg ┃ P99 ┃ Avg ┃ P99 ┃ Avg ┃ P99 ┃ Gen. ┃ Success┃
┃Conc. ┃ Rate ┃ RPS ┃ Lat.(s) ┃ Lat.(s) ┃ TTFT(s) ┃ TTFT(s) ┃ TPOT(s) ┃ TPOT(… ┃ toks/s ┃ Rate┃
┡━━━━━━╇━━━━━━╇━━━━━━╇━━━━━━━━━╇━━━━━━━━━╇━━━━━━━━━╇━━━━━━━━━╇━━━━━━━━━╇━━━━━━━━╇━━━━━━━━━╇━━━━━━━━┩
│ 10 │ INF │ 4.32 │ 2.289 │ 3.541 │ 0.041 │ 0.072 │ 0.009 │ 0.011 │ 1103.48 │ 100.0%│
└──────┴──────┴──────┴─────────┴─────────┴─────────┴─────────┴─────────┴────────┴─────────┴────────┘
Request Metrics
┏━━━━━━┳━━━━━━━━━━┳━━━━━━━━━━━━━┳━━━━━━━━━━━━┳━━━━━━━━━━━━━┳━━━━━━━━━━━━┳━━━━━━━━━━━━━┳━━━━━━━━━━━━┓
┃ ┃ ┃ ┃ P99 In ┃ Avg Out ┃ P99 Out ┃ Avg ┃ Approx┃
┃Conc. ┃ Num Reqs ┃ Avg In Toks ┃ Toks ┃ Toks ┃ Toks ┃ Turns/Req ┃ Cache Hit┃
│ 10 │ 20 │ 315.4 │ 650.0 │ 92.0 │ 128.0 │ 1.60 │ 58.1%│
└──────┴──────────┴─────────────┴────────────┴─────────────┴────────────┴─────────────┴────────────┘
指标解读:
Avg Turns/Req: 1.60:测试期间每次请求平均携带 1.60 轮上下文,符合--min-turns 2 --max-turns 5的随机采样分布(第 1 轮无历史,后续轮次历史逐步增长)。Approx Cache Hit: 58.1%:约 58% 的输入 token 来自历史对话。
3. 使用 custom_multi_turn(自定义本地对话)#
适用场景:已有 OpenAI messages 格式的对话数据,直接用于多轮压测,无需转换格式。
首先准备 JSONL 数据文件(每行一条对话):
[{"role": "user", "content": "你好"}, {"role": "assistant", "content": "你好!有什么可以帮助你?"}, {"role": "user", "content": "帮我写一首诗"}, {"role": "assistant", "content": "好的,..."}, {"role": "user", "content": "再写一首"}]
[{"role": "user", "content": "介绍一下北京"}, {"role": "assistant", "content": "北京是中国的首都..."}, {"role": "user", "content": "有哪些著名景点?"}]
然后运行压测:
evalscope perf \
--model Qwen2.5-7B-Instruct \
--url http://127.0.0.1:8801/v1/chat/completions \
--api openai \
--dataset custom_multi_turn \
--dataset-path /path/to/my_conversations.jsonl \
--max-tokens 512 \
--multi-turn \
--max-turns 3 \
--number 100 \
--parallel 10