Help us improve
Share bugs, ideas, or general feedback.
From ai-infra-auto-driven-skills
Compares LLM serving frameworks (SGLang, vLLM, TensorRT-LLM) to find optimal deployment commands under given workload, GPU budget, and latency SLA.
npx claudepluginhub bbuf/ai-infra-auto-driven-skills --plugin ai-infra-auto-driven-skillsHow this skill is triggered — by the user, by Claude, or both
Slash command
/ai-infra-auto-driven-skills:llm-serving-auto-benchmarkThe summary Claude sees in its skill listing — used to decide when to auto-load this skill
Use this skill to compare LLM serving frameworks such as SGLang, vLLM, and
configs/cookbook-llm/README.mdconfigs/cookbook-llm/deepseek-math-v2.yamlconfigs/cookbook-llm/deepseek-r1-0528.yamlconfigs/cookbook-llm/deepseek-v3.1.yamlconfigs/cookbook-llm/deepseek-v3.2.yamlconfigs/cookbook-llm/deepseek-v3.yamlconfigs/cookbook-llm/devstral-small-2-24b-instruct-2512.yamlconfigs/cookbook-llm/ernie-4.5-21b-a3b-pt.yamlconfigs/cookbook-llm/glm-4.5.yamlconfigs/cookbook-llm/glm-4.6.yamlconfigs/cookbook-llm/glm-4.7-flash.yamlconfigs/cookbook-llm/glm-4.7.yamlconfigs/cookbook-llm/glm-5-fp8.yamlconfigs/cookbook-llm/glyph.yamlconfigs/cookbook-llm/gpt-oss-120b.yamlconfigs/cookbook-llm/intern-s1.yamlconfigs/cookbook-llm/kimi-k2-instruct.yamlconfigs/cookbook-llm/kimi-k2.5.yamlconfigs/cookbook-llm/kimi-k2.6.yamlconfigs/cookbook-llm/kimi-linear-48b-a3b-instruct.yamlProvides patterns for LLM inference infrastructure with serving frameworks like vLLM, TGI, TensorRT-LLM; quantization, batching strategies, KV cache, and streaming responses. Use for optimizing latency and scaling deployments.
Interactive benchmark orchestrator for vLLM inference services. Runs single/multi-case online benchmarks, aggregates results, and auto-optimizes concurrency under latency SLOs.
Autonomously optimizes SGLang serving performance for a given LLM model by benchmarking against vLLM/TensorRT-LLL, then iteratively profiling bottlenecks, patching SGLang code, and revalidating until SGLang matches or beats the best framework under the same workload.
Share bugs, ideas, or general feedback.
Use this skill to compare LLM serving frameworks such as SGLang, vLLM, and TensorRT-LLM for the same model and workload.
Use a config-driven workflow:
base_server_flagssearch_spacesearch_space, with the baseline
candidate included firstFor model-specific starting points, prefer the shipped configs in
configs/cookbook-llm/. They define a framework-neutral LLM serving cookbook
model set and translate each entry into framework-native SGLang, vLLM, and
TensorRT-LLM server flags. Validate those configs before a real run:
python skills/llm-serving-auto-benchmark/scripts/validate_cookbook_configs.py \
skills/llm-serving-auto-benchmark/configs/cookbook-llm
If you have captured target-environment --help files, add
--help-dir <artifact-help-dir>. That check only loads configs, verifies the
server flag names, and renders candidate commands; it does not launch model
servers.
Prefer native tooling when it gives better coverage:
python -m sglang.auto_benchmark when available, otherwise
python -m sglang.bench_servingvllm bench sweep serve for server-parameter sweeps, otherwise
vllm serve plus vllm bench servetrtllm-serve for the OpenAI-compatible server plus the
TensorRT-LLM serving benchmark client or a common OpenAI-compatible benchmark
clientTensorRT-LLM has one hard scope rule in this skill: the server backend is fixed
to trtllm-serve serve --backend pytorch. Do not search TensorRT-LLM backend
choice. If a request, config, or candidate asks for trt, an engine backend, or
any other non-PyTorch TensorRT-LLM server backend, reject that candidate as
unsupported for this skill and record the reason. This does not change the
benchmark client backend; the TensorRT-LLM benchmark client still uses
OpenAI-compatible modes such as --backend openai or --backend openai-chat.
Only pick a winner after each requested framework has had its main serving knobs tuned.
The parameter lists in this skill are not a compatibility contract. They are
version-sensitive candidate knob families. Before every real run, record the
exact framework version or git commit and verify the concrete CLI flag names
with --help in the target environment.
The default search style is framework-neutral: start from a mostly pure-TP baseline, sweep a small set of high-impact runtime knobs, and cap the first pass around 10 candidates per framework. Do not search memory fractions by default.
This skill is target-agnostic. It assumes any one of the following is available, and nothing more:
ssh <host> with the framework images already
running in a container there;Do not assume a specific operator host name inside this skill's own workflow.
The concrete SSH wiring, container names, workspace paths, and HF token plumbing
for a given box live in operator-side per-host skills; this skill only requires
that the caller can reach a shell inside a container with sglang, vllm, or
tensorrt_llm installed.
Reference files are optional and version-sensitive. Treat historical flag notes as evidence from one image, not as a compatibility guarantee for the next run.
Additional H100 validation on 2026-05-01 used two 2-card models with a
bounded search of two SGLang memory-fraction candidates and two vLLM
memory-utilization candidates. The workload was random input 512, output
64, 8 prompts, and 2 warmup requests, only to prove the search and summary
path can finish quickly.
| Model | GPUs | Best SGLang | Best vLLM | Artifact root |
|---|---|---|---|---|
Qwen/Qwen3-8B | 2x H100, TP=2 | sglang_mem086, 21.64 req/s, 1385.05 output tok/s, mean TTFT 70.54 ms | vllm_mem080, 22.88 req/s, 1464.25 output tok/s, mean TTFT 60.56 ms | /data/bbuf/validate/core_skill_validation_20260501/qwen3_8b/auto_benchmark |
mistralai/Mistral-7B-Instruct-v0.3 | 2x H100, TP=2 | sglang_mem080, 24.09 req/s, 1541.92 output tok/s, mean TTFT 61.47 ms | vllm_mem090, 24.76 req/s, 1584.54 output tok/s, mean TTFT 58.63 ms | /data/bbuf/validate/core_skill_validation_20260501/mistral_7b_instruct_v03/auto_benchmark |
This skill is a playbook plus a config+validator toolchain, not a turn-key orchestrator. The operator still launches servers, drives workloads, and writes one normalized JSONL row per candidate.
The scripts/ directory contains exactly two tools:
validate_cookbook_configs.py: load cookbook YAML, render bounded candidate
server commands, and check flag names against captured --help snapshots
without launching servers.compare_benchmark_results.py: turn normalized per-candidate JSONL into the
markdown and optional CSV tables described in the Output Contract.Cookbook configs under configs/cookbook-llm/ must pass the validator. The
shorter references/example-plan.yaml is a
one-off runtime-plan skeleton and is not expected to pass as-is. Use
references/result-schema.md as the single source
of truth for SLA key names.
Collect these before a long run:
--help snapshots, and whether each search parameter was
accepted by that exact CLIIf real production traffic is the goal, use the real request distribution. A synthetic workload is fine for bring-up and first-pass comparison, but it is not enough for a production choice.
Record each scenario's input/output length distribution in the normalized
result rows. This is now part of the profiler handoff contract: if SGLang is
slower and sglang-sota-humanize-loop invokes llm-torch-profiler-analysis,
the profiler workload must reuse the slow SGLang benchmark scenario lengths
instead of falling back to its generic prefill 4090->1 and decode 1->2048
defaults.
Short list of failure modes that have bitten past validation runs. Check these before starting a long sweep.
fa3 attention backends need Hopper or newer. On A100, L40S, RTX
5090, and older GPUs, drop fa3 from the SGLang search_space and keep
flashinfer (or triton when FlashInfer is unavailable).bench_serving has two SGLang-facing backends: --backend sglang for
the native /generate endpoint and --backend sglang-oai for the
OpenAI-compatible endpoint. For cross-framework comparisons, prefer
sglang-oai so every framework is measured on the same request path.--enable-dbo only works when the target vLLM image is built with a
supported all2all backend. Keep DBO out of the default candidate list unless
the operator has verified the image.--max-num-partial-prefills > 1 is model- and runtime-gated. Keep 1
in the default pass; raise only after a preflight with the actual model.7021547 on 2026-05-15, and
the serving flag evidence from b9e1945 still applies because newer commits
only touched CI/test-waive files. That evidence accepts both
--kv_cache_free_gpu_memory_fraction and --free_gpu_memory_fraction as
server aliases. Keep
kv_cache_free_gpu_memory_fraction in shipped configs because it maps
directly to the KvCacheConfig field and remains compatible with the older
validation image that rejected the shorter alias.--ipc=host, --ulimit memlock=-1, --ulimit stack=67108864,
--shm-size=16g, and NCCL_IB_DISABLE=1 (for single-node) or an equivalent
NCCL setup. Keep these as a starting point, not as a version-independent
requirement.openai and openai-chat, not trtllm. This is separate from the server
backend, which is pinned to pytorch by this skill.trtllm benchmark_serving --dataset-name random silently falls back to
ShareGPT sampling without --random-ids (or --download-path).max_seq_len / max_model_len / context_length candidates must cover
max(input_len + output_len) across every scenario, including values inside
search_space, not just the baseline. The validator checks this; do not
bypass it.HF_TOKEN, HUGGINGFACE_HUB_TOKEN, or any upstream API key into
a saved artifact. Pass them through container -e VAR (unquoted on the right
side so the host value is inherited) and keep them out of server_command
and benchmark_command fields written to the result JSONL.Use these rules throughout the benchmark:
Verify all requested frameworks before starting a search:
python -m sglang.launch_server --help
python -m sglang.bench_serving --help
vllm serve --help
vllm serve --help=all
vllm bench serve --help
vllm bench serve --help=all
vllm bench sweep serve --help=all
trtllm-serve serve --help
python -m tensorrt_llm.serve.scripts.benchmark_serving --help
Use the framework-specific --help output in the target environment as the
source of truth. Do not keep a stale launch flag just because it appears in an
old note.
vLLM 0.19 and newer use grouped help. Plain vllm serve --help only shows the
groups, so capture --help=all before deciding whether a search knob exists.
Save these --help outputs into the run artifact directory. If a listed search
knob is missing from the current CLI, remove or translate that knob before
running the benchmark. Do not silently pass unknown flags.
For TensorRT-LLM, also confirm that trtllm-serve serve --help accepts
--backend pytorch. If it does not, mark TensorRT-LLM unsupported in that
environment rather than falling back to a different server backend.
For each framework, launch a minimal server, confirm /v1/models or the native
model-info endpoint, send one streaming request, run one tiny benchmark with at
least 5 requests, then save the launch command, benchmark command, server log,
and benchmark output.
Before any GPU-backed smoke run, check the requested GPU ids directly with
nvidia-smi. If a requested GPU is already in use, stop and record that fact.
Do not silently borrow a different GPU count for a performance comparison. It is
fine to run a smaller one-GPU smoke only when the result is clearly labeled as a
flow check rather than a fair throughput comparison.
If the target environment runs through containers, follow references/container-runbook.md and save image tags, pull commands, launch/benchmark logs, and cleanup commands.
Use one canonical workload for all frameworks. Recommended JSONL row shape:
{"prompt": [{"role": "user", "content": "Summarize this text."}], "output_len": 256}
{"prompt": "Write a short explanation of CUDA graphs.", "output_len": 128}
Optional fields:
{
"prompt": [{"role": "user", "content": "Use low temperature."}],
"output_len": 256,
"extra_request_body": {"temperature": 0.0, "top_p": 0.95},
"metadata": {"source": "prod-sample"}
}
When converting user data:
extra_request_bodyFor synthetic bring-up, use the shipped two-scenario shape:
dataset:
kind: random
num_prompts: 80
scenario_names: [chat, summarization]
input_len: [1000, 8000]
output_len: [1000, 1000]
Each aligned input_len / output_len pair is one scenario. Do not take the
cartesian product unless the user asks for that.
Name each scenario and keep the aligned pair in the artifacts. For custom
datasets, compute or record representative input_len and output_len
buckets, at least p50 and p95 when possible, so later profiler runs can match
the slow bucket rather than profiling an unrelated synthetic shape.
Before searching any sequence-length limit, compute the largest
input_len + output_len in the dataset. SGLang context_length, vLLM
max_model_len, and TensorRT-LLM max_seq_len must be at least that value for
every candidate that is expected to run all scenarios.
Use the smallest tier that can answer the user's question:
Default budget:
num_prompts: 80 for the default cross-framework comparison; num_prompts: 20 per scenario is acceptable for a smoke/flow check and must be labeled as
such in the artifact (not as a performance result).search.max_candidates_per_framework: 10 for the first useful passsearch_spaceKeep these in base_server_flags unless the user specifically wants a capacity
or memory study:
mem_fraction_staticschedule_policygpu_memory_utilizationkv_cache_free_gpu_memory_fractionThese are real knobs, but they widen the search quickly and often turn a serving comparison into a memory-limit study.
Prefer the SGLang auto-benchmark runner when the target checkout supports it:
python -m sglang.auto_benchmark run --config /path/to/sglang.yaml
Otherwise launch the server manually and benchmark with:
python -m sglang.bench_serving \
--backend sglang \
--dataset-name random \
--random-input-len 1024 \
--random-output-len 256 \
--num-prompts 80 \
--request-rate 8 \
--output-file /path/to/sglang/results.json \
--output-details
Version-sensitive SGLang knob families to verify:
tp_size, pp_size, dp_size, ep_sizeattention_backend, prefill_attention_backend, decode_attention_backendsampling_backendmax_running_requests, max_queued_requestschunked_prefill_size, prefill_max_requests, max_prefill_tokensmax_total_tokens, page_sizeKeep mem_fraction_static and schedule_policy pinned in the default pass,
matching the shared cookbook config style.
For quick smoke tests, it is reasonable to disable CUDA graph and piecewise CUDA graph startup work if the goal is only to prove the framework flow. Record those flags in the artifact. Do not carry that smoke setting into a performance winner unless the user asked to tune eager-mode serving.
Use vLLM's sweep runner when available:
vllm bench sweep serve \
--serve-cmd 'vllm serve <model> --port 8000' \
--bench-cmd 'vllm bench serve --backend vllm --model <model> --port 8000 --dataset-name random --num-prompts 80' \
--serve-params /path/to/vllm_serve_params.json \
--bench-params /path/to/vllm_bench_params.json \
--output-dir /path/to/vllm_results
If sweep support is unavailable, run vllm serve for each candidate and measure
with vllm bench serve.
Version-sensitive vLLM knob families to verify:
gpu_memory_utilizationmax_num_seqsmax_num_batched_tokensmax_model_lenenable_chunked_prefill, partial prefill limits, and DBO thresholdsvLLM should get a normal sweep, not one baseline command. See
references/framework-reference.md for
native command templates and cross-framework knob families. Confirm each flag on
the target image's --help before a run.
Keep gpu_memory_utilization in the baseline for the default pass. Search it
only when the question is explicitly about fitting the model or trading capacity
against throughput.
Keep DBO and all2all backend settings out of the default pass unless the target
vLLM environment is already set up for them. They are real tuning knobs, but a
candidate can fail at startup if the required all2all backend is not available.
Also preflight concurrent partial prefill before raising
max_num_partial_prefills above 1; some model/runtime combinations reject it at
startup.
Use trtllm-serve serve as the server entrypoint when the target environment
supports it:
trtllm-serve serve <model> \
--backend pytorch \
--tp_size <tp> \
--pp_size <pp> \
--kv_cache_free_gpu_memory_fraction 0.75 \
--host 0.0.0.0 \
--port 8000
Then benchmark the OpenAI-compatible endpoint with the TensorRT-LLM serving benchmark client or with the same OpenAI-compatible client used for the other frameworks.
In the historical TensorRT-LLM 1.0.0 validation image,
benchmark_serving --dataset-name random sampled from ShareGPT unless either
--download-path or --random-ids was passed. For a fast synthetic smoke test,
pass --random-ids, then confirm the behavior on the target TensorRT-LLM image.
TensorRT-LLM flag names are especially version-sensitive. In the validated
TensorRT-LLM 1.0.0 image, the KV-cache memory flag accepted by
trtllm-serve serve was --kv_cache_free_gpu_memory_fraction, not
--free_gpu_memory_fraction. Current mainline was rechecked at 7021547 on
2026-05-15; the newer commits did not touch serving code, so the b9e1945
serving flag evidence still applies: both aliases are accepted and
trtllm-serve still defaults to the PyTorch backend. Always verify flags with
trtllm-serve serve --help before running a search on any GPU target.
TensorRT-LLM backend policy for this skill:
--backend pytorchbackend: pytorch in base_server_flagsbackend to search_spacetrt, engine-backed serving, or any other non-PyTorch TensorRT-LLM
server backend as unsupported for this skillVersion-sensitive TensorRT-LLM knob families to verify:
tp_size, pp_size, and ep_sizetrtllm-serve with the PyTorch backendThe trtllm-serve serve CLI exposes fewer direct runtime knobs than SGLang or
vLLM. Use direct flags when they exist, then use --extra_llm_api_options for
PyTorch-backend settings that are not top-level CLI flags. Keep unsupported
backend or engine requests in the failure table instead of translating them.
Keep kv_cache_free_gpu_memory_fraction in the baseline for the default pass.
Search max_batch_size, max_num_tokens, max_seq_len, and validated
PyTorch-backend config options first. The server backend remains fixed to
pytorch.
Write one JSONL row per candidate using the schema in references/result-schema.md. Then run:
python skills/llm-serving-auto-benchmark/scripts/compare_benchmark_results.py \
--input /path/to/candidates.jsonl \
--output /path/to/summary.md
Rank candidates in this order:
Keep the SLA gate itself unchanged. In the cookbook configs and normalized
result schema, TTFT SLA still uses max_p50_ttft_ms and TPOT SLA still uses
max_p50_tpot_ms; only the default cross-candidate comparison order switches
to p50 TTFT and p50 TPOT.
Return a compact report with workload/SLA, hardware and framework versions, best deployment-command tables per framework/scenario, one cross-framework comparison table, exact launch and benchmark commands for winners, and artifact paths for workload, raw/normalized results, CSV or markdown summary, and server logs.
When SGLang is not the winner, include a profiler handoff note with the slow
SGLang scenario name and the exact input/output lengths or percentile bucket to
pass to llm-torch-profiler-analysis.
Include failed or excluded candidates with reasons. Explain that this table is a record of tried configs that were not selected: candidates that failed, were skipped by policy, or completed but missed the SLA. Add caveats for synthetic workloads, incomplete fair searches, or framework-specific parameter substitutions.
Use references/framework-reference.md when you need command templates, source links, or knob-family mappings. Use references/example-plan.yaml as the starting point for a full cross-framework run plan.