| title | Metrics |
|---|
Dynamo provides built-in metrics capabilities through the Dynamo metrics API, which is automatically available whenever you use the DistributedRuntime framework. This document serves as a reference for all available metrics in Dynamo.
For visualization setup instructions, see the Prometheus and Grafana Setup Guide.
For creating custom metrics, see the Metrics Developer Guide.
| Variable | Description | Default | Example |
|---|---|---|---|
DYN_SYSTEM_PORT |
Backend component metrics/health port | -1 (disabled) |
8081 |
DYN_HTTP_PORT |
Frontend HTTP port (also configurable via --http-port flag) |
8000 |
8000 |
NIXL_TELEMETRY_ENABLE |
Enable NIXL telemetry (see NIXL Telemetry Metrics). Options: y, n |
n (disabled) |
y |
This is a single machine example.
For visualizing metrics with Prometheus and Grafana, start the observability stack. See Observability Getting Started for instructions.
Launch a frontend and vLLM backend to test metrics:
# Start frontend (default port 8000, override with --http-port or DYN_HTTP_PORT env var)
$ python -m dynamo.frontend
# Enable backend worker's system metrics on port 8081
$ DYN_SYSTEM_PORT=8081 python -m dynamo.vllm --model Qwen/Qwen3-0.6B \
--enforce-eager --no-enable-prefix-caching --max-num-seqs 3Wait for the vLLM worker to start, then send requests and check metrics:
# Send a request
curl -H 'Content-Type: application/json' \
-d '{
"model": "Qwen/Qwen3-0.6B",
"max_completion_tokens": 100,
"messages": [{"role": "user", "content": "Hello"}]
}' \
http://localhost:8000/v1/chat/completions
# Check metrics from the backend worker
curl -s localhost:8081/metrics | grep dynamo_componentDynamo exposes metrics in Prometheus Exposition Format text at the /metrics HTTP endpoint. All Dynamo-generated metrics use the dynamo_* prefix and include labels (dynamo_namespace, dynamo_component, dynamo_endpoint) to identify the source component.
Example Prometheus Exposition Format text:
# HELP dynamo_component_requests_total Total requests processed
# TYPE dynamo_component_requests_total counter
dynamo_component_requests_total{dynamo_namespace="default",dynamo_component="worker",dynamo_endpoint="generate"} 42
# HELP dynamo_component_request_duration_seconds Request processing time
# TYPE dynamo_component_request_duration_seconds histogram
dynamo_component_request_duration_seconds_bucket{dynamo_namespace="default",dynamo_component="worker",dynamo_endpoint="generate",le="0.005"} 10
dynamo_component_request_duration_seconds_bucket{dynamo_namespace="default",dynamo_component="worker",dynamo_endpoint="generate",le="0.01"} 15
dynamo_component_request_duration_seconds_bucket{dynamo_namespace="default",dynamo_component="worker",dynamo_endpoint="generate",le="+Inf"} 42
dynamo_component_request_duration_seconds_sum{dynamo_namespace="default",dynamo_component="worker",dynamo_endpoint="generate"} 2.5
dynamo_component_request_duration_seconds_count{dynamo_namespace="default",dynamo_component="worker",dynamo_endpoint="generate"} 42
Dynamo exposes several categories of metrics:
- Frontend Metrics (
dynamo_frontend_*) - Request handling, token processing, and latency measurements - Component Metrics (
dynamo_component_*) - Request counts, processing times, byte transfers, and system uptime - Specialized Component Metrics (e.g.,
dynamo_preprocessor_*) - Component-specific metrics - Engine Metrics (Pass-through) - Backend engines expose their own metrics: vLLM (
vllm:*), SGLang (sglang:*), TensorRT-LLM (trtllm_*)
The Dynamo metrics API is available on DistributedRuntime, Namespace, Component, and Endpoint, providing a hierarchical approach to metric collection that matches Dynamo's distributed architecture:
DistributedRuntime: Global metrics across the entire runtimeNamespace: Metrics scoped to a specific dynamo_namespaceComponent: Metrics for a specific dynamo_component within a namespaceEndpoint: Metrics for individual dynamo_endpoint within a component
This hierarchical structure allows you to create metrics at the appropriate level of granularity for your monitoring needs.
Backend workers (python -m dynamo.vllm, python -m dynamo.sglang, etc.) expose dynamo_component_* metrics on the system status port (configurable via DYN_SYSTEM_PORT, disabled by default). In Kubernetes the operator typically sets DYN_SYSTEM_PORT=9090; for local development you must set it explicitly (e.g. DYN_SYSTEM_PORT=8081).
The core Dynamo backend system exposes metrics at the /metrics endpoint with the dynamo_component_* prefix for all components that use the DistributedRuntime framework:
dynamo_component_inflight_requests: Requests currently being processed (gauge)dynamo_component_request_bytes_total: Total bytes received in requests (counter)dynamo_component_request_duration_seconds: Request processing time (histogram)dynamo_component_requests_total: Total requests processed (counter)dynamo_component_response_bytes_total: Total bytes sent in responses (counter)dynamo_component_uptime_seconds: DistributedRuntime uptime (gauge). Automatically updated before each Prometheus scrape on both the frontend (/metricson port 8000) and the system status server (/metricsonDYN_SYSTEM_PORTwhen set).
Access backend component metrics:
# Set DYN_SYSTEM_PORT to enable the system status server
DYN_SYSTEM_PORT=8081 python -m dynamo.vllm --model <model>
curl http://localhost:8081/metricsSome components expose additional metrics specific to their functionality:
dynamo_preprocessor_*: Metrics specific to preprocessor components
Important: The frontend and backend workers are separate components that expose metrics on different ports. See Backend Component Metrics for backend metrics.
The Dynamo HTTP Frontend (python -m dynamo.frontend) exposes dynamo_frontend_* metrics on port 8000 by default (configurable via --http-port or DYN_HTTP_PORT) at the /metrics endpoint. Most metrics include model labels containing the model name:
dynamo_frontend_inflight_requests: Inflight requests (gauge)dynamo_frontend_queued_requests: Number of requests in HTTP processing queue (gauge)dynamo_frontend_disconnected_clients: Number of disconnected clients (gauge)dynamo_frontend_input_sequence_tokens: Input sequence length (histogram)dynamo_frontend_cached_tokens: Number of cached tokens (prefix cache hits) per request (histogram)dynamo_frontend_inter_token_latency_seconds: Inter-token latency (histogram)dynamo_frontend_output_sequence_tokens: Output sequence length (histogram)dynamo_frontend_output_tokens_total: Total number of output tokens generated (counter)dynamo_frontend_request_duration_seconds: LLM request duration (histogram)dynamo_frontend_requests_total: Total LLM requests (counter)dynamo_frontend_time_to_first_token_seconds: Time to first token (histogram)dynamo_frontend_model_migration_total: Total number of request migrations due to worker unavailability (counter, labels:model,migration_type)
Access frontend metrics:
curl http://localhost:8000/metricsNote: The dynamo_frontend_inflight_requests metric tracks requests from HTTP handler start until the complete response is finished, while dynamo_frontend_queued_requests tracks requests from HTTP handler start until first token generation begins (including prefill time). HTTP queue time is a subset of inflight time.
The frontend also exposes model configuration metrics (on port 8000 /metrics endpoint) with the dynamo_frontend_model_* prefix. These metrics are populated from the worker backend registration service when workers register with the system. All model configuration metrics include a model label.
Runtime Config Metrics (from ModelRuntimeConfig): These metrics come from the runtime configuration provided by worker backends during registration.
dynamo_frontend_model_total_kv_blocks: Total KV blocks available for a worker serving the model (gauge)dynamo_frontend_model_max_num_seqs: Maximum number of sequences for a worker serving the model (gauge)dynamo_frontend_model_max_num_batched_tokens: Maximum number of batched tokens for a worker serving the model (gauge)
MDC Metrics (from ModelDeploymentCard): These metrics come from the Model Deployment Card information provided by worker backends during registration. Note that when multiple worker instances register with the same model name, only the first instance's configuration metrics (runtime config and MDC metrics) will be populated. Subsequent instances with duplicate model names will be skipped for configuration metric updates.
dynamo_frontend_model_context_length: Maximum context length for a worker serving the model (gauge)dynamo_frontend_model_kv_cache_block_size: KV cache block size for a worker serving the model (gauge)dynamo_frontend_model_migration_limit: Request migration limit for a worker serving the model (gauge)
This section explains the distinction between two key metrics used to track request processing:
- Inflight: Tracks requests from HTTP handler start until the complete response is finished
- HTTP Queue: Tracks requests from HTTP handler start until first token generation begins (including prefill time)
Example Request Flow:
curl -s localhost:8000/v1/completions -H "Content-Type: application/json" -d '{
"model": "Qwen/Qwen3-0.6B",
"prompt": "Hello let's talk about LLMs",
"stream": false,
"max_tokens": 1000
}'
Timeline:
sequenceDiagram
participant Client
participant Frontend as Frontend:8000
participant Backend as Backend (SGLang/TRT/vLLM)
Client->>Frontend: Request start
Note over Frontend,Backend: HTTP queue begins
Frontend->>Backend: Forward request
Note over Backend: Start prefill
Backend-->>Frontend: First token
Note over Frontend,Backend: HTTP queue ends
loop Token generation
Backend-->>Frontend: Tokens
end
Backend-->>Frontend: Last token
Frontend-->>Client: Complete response
Note over Frontend: Inflight ends
Concurrency Example: Suppose the backend allows 3 concurrent requests and there are 10 clients continuously hitting the frontend:
- All 10 requests will be counted as inflight (from start until complete response)
- 7 requests will be in HTTP queue most of the time
- 3 requests will be actively processed (between first token and last token)
Key Differences:
- Inflight: Measures total request lifetime including processing time
- HTTP Queue: Measures queuing time before processing begins (including prefill time)
- HTTP Queue ≤ Inflight (HTTP queue is a subset of inflight time)
The router exposes metrics for monitoring routing decisions and overhead. Defined in lib/llm/src/kv_router/metrics.rs.
For router deployment modes, see the Router Guide. For router flags and tuning, see Configuration and Tuning.
Not all metrics appear in every deployment. The chart below shows which metric groups are registered and populated in each configuration:
| Metric Group | Frontend + KV (agg) | Frontend + KV (disagg) | Frontend + non-KV (round-robin/random/direct) | Standalone Router |
|---|---|---|---|---|
dynamo_component_router_* (request metrics) |
Registered and populated | Registered and populated | Registered, always zero | Populated (on DYN_SYSTEM_PORT) |
dynamo_router_overhead_* (routing overhead) |
Registered and populated | Registered and populated | Not registered | Not created |
dynamo_frontend_router_queue_* (queue depth) |
Registered; populated when --router-queue-threshold set |
Registered; populated when --router-queue-threshold set |
Not registered | Not created |
dynamo_component_kv_cache_events_applied (indexer) |
Populated when KV events are received | Populated when KV events are received | Not registered | Populated when KV events are received |
dynamo_frontend_worker_* (per-worker load/timing) |
Registered and populated | Registered and populated (worker_type=prefill/decode) |
Registered and populated (worker_type=decode) |
Not created |
Key:
- Registered and populated: Metric appears at
/metricswith real values - Registered, always zero: Metric appears at
/metricsbut the counter/histogram is never incremented (useful for dashboards that expect the metric to exist) - Not registered / Not created: Metric does not appear at
/metricsat all
Scrape endpoints:
- Frontend:
/metricson HTTP port (default 8000, configurable via--http-portorDYN_HTTP_PORT) - Standalone router:
/metricsonDYN_SYSTEM_PORT(must be set explicitly; default is-1/ disabled) - Backend workers:
/metricsonDYN_SYSTEM_PORT(separate from frontend metrics)
Histograms and counters for aggregate request-level statistics. Eagerly registered via from_component() with the DRT MetricsRegistry hierarchy. On the frontend, exposed at /metrics on the HTTP port (default 8000) via the drt_metrics bridge. On the standalone router (python -m dynamo.router), exposed on DYN_SYSTEM_PORT when set. Populated per-request when --router-mode kv is active; registered with zero values in non-KV modes.
All metrics carry the standard hierarchy labels (dynamo_namespace, dynamo_component, dynamo_endpoint).
| Metric | Type | Description |
|---|---|---|
dynamo_component_router_requests_total |
Counter | Total requests processed by the router |
dynamo_component_router_time_to_first_token_seconds |
Histogram | Time to first token (seconds) |
dynamo_component_router_inter_token_latency_seconds |
Histogram | Average inter-token latency (seconds) |
dynamo_component_router_input_sequence_tokens |
Histogram | Input sequence length (tokens) |
dynamo_component_router_output_sequence_tokens |
Histogram | Output sequence length (tokens) |
dynamo_component_router_kv_hit_rate |
Histogram | Predicted KV cache hit rate at routing time (0.0-1.0) |
Histograms (in milliseconds) tracking the time spent in each phase of the routing decision for every request. Registered on the frontend port (default 8000) at /metrics with a router_id label (the frontend's discovery instance ID). These metrics are only created when the frontend has DRT discovery enabled (i.e., --router-mode kv); they do not appear in non-KV modes or on the standalone router.
| Metric | Type | Description |
|---|---|---|
dynamo_router_overhead_block_hashing_ms |
Histogram | Time computing block hashes |
dynamo_router_overhead_indexer_find_matches_ms |
Histogram | Time in indexer find_matches |
dynamo_router_overhead_seq_hashing_ms |
Histogram | Time computing sequence hashes |
dynamo_router_overhead_scheduling_ms |
Histogram | Time in scheduler worker selection |
dynamo_router_overhead_total_ms |
Histogram | Total routing overhead per request |
Gauge tracking the number of requests pending in the router's scheduler queue. Only registered when --router-queue-threshold is set. Labeled by worker_type to distinguish prefill vs. decode queues in disaggregated mode.
| Metric | Type | Description |
|---|---|---|
dynamo_frontend_router_queue_pending_requests |
Gauge | Requests pending in the router scheduler queue |
Labels: worker_type (prefill or decode)
Tracks KV cache events applied to the router's radix tree index. Only appears when --router-kv-overlap-score-weight is greater than 0 (default) and workers are publishing KV events. Will not appear if --router-kv-overlap-score-weight 0 is set or no KV events have been received.
| Metric | Type | Description |
|---|---|---|
dynamo_component_kv_cache_events_applied |
Counter | KV cache events applied to the index |
Additional labels: status (ok / parent_block_not_found / block_not_found / invalid_block), event_type (stored / removed / cleared)
These appear once workers register and begin serving requests. They are registered on the frontend's local Prometheus registry (not component-scoped) and do not carry dynamo_namespace or dynamo_component labels. These metrics are frontend-only and are not available on the standalone router.
| Metric | Type | Description |
|---|---|---|
dynamo_frontend_worker_active_decode_blocks |
Gauge | Active KV cache decode blocks per worker |
dynamo_frontend_worker_active_prefill_tokens |
Gauge | Active prefill tokens queued per worker |
dynamo_frontend_worker_last_time_to_first_token_seconds |
Gauge | Last observed TTFT per worker (seconds) |
dynamo_frontend_worker_last_input_sequence_tokens |
Gauge | Last observed input sequence length per worker |
dynamo_frontend_worker_last_inter_token_latency_seconds |
Gauge | Last observed ITL per worker (seconds) |
Labels:
| Label | Example Value | Description |
|---|---|---|
worker_id |
7890 |
Worker instance ID (etcd lease ID) |
dp_rank |
0 |
Data-parallel rank |
worker_type |
prefill or decode |
Worker role |
In disaggregated mode, the worker_type label shows both "prefill" and "decode" values; in aggregated mode, all workers report as "decode".
NIXL exposes its own Prometheus metrics on a separate port from Dynamo metrics. These metrics track KV cache and embedding data transfers and are only populated during disaggregated serving or multimodal embedding transfers.
To enable, set these environment variables on your worker process:
# Prefill worker
NIXL_TELEMETRY_ENABLE=y NIXL_TELEMETRY_EXPORTER=prometheus \
NIXL_TELEMETRY_PROMETHEUS_PORT=19090 DYN_SYSTEM_PORT=8081 \
python -m dynamo.vllm --model <model> --disaggregation-mode prefill
# Decode worker (different NIXL port to avoid collision)
NIXL_TELEMETRY_ENABLE=y NIXL_TELEMETRY_EXPORTER=prometheus \
NIXL_TELEMETRY_PROMETHEUS_PORT=19091 DYN_SYSTEM_PORT=8082 \
python -m dynamo.vllm --model <model> --disaggregation-mode decode
# Scrape NIXL metrics (separate from Dynamo metrics on 8081/8082)
curl http://localhost:19090/metricsFor the full list of metrics, configuration options, and architecture details, see the upstream NIXL Telemetry documentation and Prometheus exporter README. For Kubernetes, see Enable NIXL Telemetry.