loggers.py 20.2 KB
Newer Older
1
2
# SPDX-License-Identifier: Apache-2.0

3
import logging
4
5
import time
from abc import ABC, abstractmethod
6
from typing import Callable, Optional
7

8
import numpy as np
9
import prometheus_client
10

11
from vllm.config import SupportsMetricsInfo, VllmConfig
12
from vllm.logger import init_logger
13
from vllm.v1.core.kv_cache_utils import PrefixCachingMetrics
14
from vllm.v1.engine import FinishReason
15
from vllm.v1.metrics.stats import IterationStats, SchedulerStats
16
from vllm.v1.spec_decode.metrics import SpecDecodingLogging, SpecDecodingProm
17
18
19
20
21

logger = init_logger(__name__)

_LOCAL_LOGGING_INTERVAL_SEC = 5.0

22
23
StatLoggerFactory = Callable[[VllmConfig, int], "StatLoggerBase"]

24
25

class StatLoggerBase(ABC):
26
27
28
29
30
31
32
33
34
35
    """Interface for logging metrics.

    API users may define custom loggers that implement this interface.
    However, note that the `SchedulerStats` and `IterationStats` classes
    are not considered stable interfaces and may change in future versions.
    """

    @abstractmethod
    def __init__(self, vllm_config: VllmConfig, engine_index: int = 0):
        ...
36
37

    @abstractmethod
38
    def record(self, scheduler_stats: SchedulerStats,
39
               iteration_stats: Optional[IterationStats]):
40
41
        ...

42
43
44
45
    @abstractmethod
    def log_engine_initialized(self):
        ...

46
47
48
    def log(self):  # noqa
        pass

49
50
51

class LoggingStatLogger(StatLoggerBase):

52
    def __init__(self, vllm_config: VllmConfig, engine_index: int = 0):
53
        self.engine_index = engine_index
54
        self.vllm_config = vllm_config
55
        self._reset(time.monotonic())
56
        self.last_scheduler_stats = SchedulerStats()
57
58
59
        # Prefix cache metrics. This cannot be reset.
        # TODO: Make the interval configurable.
        self.prefix_caching_metrics = PrefixCachingMetrics()
60
        self.spec_decoding_logging = SpecDecodingLogging()
61
62
        self.last_prompt_throughput: float = 0.0
        self.last_generation_throughput: float = 0.0
63

64
65
66
67
    def _reset(self, now):
        self.last_log_time = now

        # Tracked stats over current local logging interval.
68
69
        self.num_prompt_tokens: list[int] = []
        self.num_generation_tokens: list[int] = []
70

71
72
73
74
75
76
    def _track_iteration_stats(self, iteration_stats: IterationStats):
        # Save tracked stats for token counters.
        self.num_prompt_tokens.append(iteration_stats.num_prompt_tokens)
        self.num_generation_tokens.append(
            iteration_stats.num_generation_tokens)

77
    def _get_throughput(self, tracked_stats: list[int], now: float) -> float:
78
79
80
        # Compute summary metrics for tracked stats
        return float(np.sum(tracked_stats) / (now - self.last_log_time))

81
    def record(self, scheduler_stats: SchedulerStats,
82
               iteration_stats: Optional[IterationStats]):
83
84
        """Log Stats to standard output."""

85
86
        if iteration_stats:
            self._track_iteration_stats(iteration_stats)
87

88
89
        self.prefix_caching_metrics.observe(scheduler_stats.prefix_cache_stats)

90
        if scheduler_stats.spec_decoding_stats is not None:
91
            self.spec_decoding_logging.observe(
92
93
                scheduler_stats.spec_decoding_stats)

94
        self.last_scheduler_stats = scheduler_stats
95

96
97
    def log(self):
        now = time.monotonic()
98
99
100
101
102
        prompt_throughput = self._get_throughput(self.num_prompt_tokens, now)
        generation_throughput = self._get_throughput(
            self.num_generation_tokens, now)

        self._reset(now)
103

104
105
        scheduler_stats = self.last_scheduler_stats

106
107
108
109
110
111
112
113
114
        log_fn = logger.info
        if not any(
            (prompt_throughput, generation_throughput,
             self.last_prompt_throughput, self.last_generation_throughput)):
            # Avoid log noise on an idle production system
            log_fn = logger.debug
        self.last_generation_throughput = generation_throughput
        self.last_prompt_throughput = prompt_throughput

115
        # Format and print output.
116
        log_fn(
117
            "Engine %03d: "
118
119
            "Avg prompt throughput: %.1f tokens/s, "
            "Avg generation throughput: %.1f tokens/s, "
120
121
122
            "Running: %d reqs, Waiting: %d reqs, "
            "GPU KV cache usage: %.1f%%, "
            "Prefix cache hit rate: %.1f%%",
123
            self.engine_index,
124
125
            prompt_throughput,
            generation_throughput,
126
127
            scheduler_stats.num_running_reqs,
            scheduler_stats.num_waiting_reqs,
128
            scheduler_stats.gpu_cache_usage * 100,
129
            self.prefix_caching_metrics.hit_rate * 100,
130
        )
131
        self.spec_decoding_logging.log(log_fn=log_fn)
132

133
134
135
136
137
138
    def log_engine_initialized(self):
        logger.info(
            "vllm cache_config_info with initialization " \
            "after num_gpu_blocks is: %d",
            self.vllm_config.cache_config.num_gpu_blocks)

139
140

class PrometheusStatLogger(StatLoggerBase):
141
142
143
144
    _gauge_cls = prometheus_client.Gauge
    _counter_cls = prometheus_client.Counter
    _histogram_cls = prometheus_client.Histogram
    _spec_decoding_cls = SpecDecodingProm
145

146
    def __init__(self, vllm_config: VllmConfig, engine_index: int = 0):
147
        self._unregister_vllm_metrics()
148
149
        self.vllm_config = vllm_config
        self.engine_index = engine_index
150
151
152
153
154
        # Use this flag to hide metrics that were deprecated in
        # a previous release and which will be removed future
        self.show_hidden_metrics = \
            vllm_config.observability_config.show_hidden_metrics

155
156
157
158
159
        labelnames = ["model_name", "engine"]
        labelvalues = [
            vllm_config.model_config.served_model_name,
            str(engine_index)
        ]
160

161
        max_model_len = vllm_config.model_config.max_model_len
162

163
        self.spec_decoding_prom = self._spec_decoding_cls(
164
165
            vllm_config.speculative_config, labelnames, labelvalues)

166
167
168
        #
        # Scheduler state
        #
169
        self.gauge_scheduler_running = self._gauge_cls(
170
171
172
173
            name="vllm:num_requests_running",
            documentation="Number of requests in model execution batches.",
            labelnames=labelnames).labels(*labelvalues)

174
        self.gauge_scheduler_waiting = self._gauge_cls(
175
176
177
178
            name="vllm:num_requests_waiting",
            documentation="Number of requests waiting to be processed.",
            labelnames=labelnames).labels(*labelvalues)

179
180
181
        #
        # GPU cache
        #
182
        self.gauge_gpu_cache_usage = self._gauge_cls(
183
184
185
186
            name="vllm:gpu_cache_usage_perc",
            documentation="GPU KV-cache usage. 1 means 100 percent usage.",
            labelnames=labelnames).labels(*labelvalues)

187
        self.counter_gpu_prefix_cache_queries = self._counter_cls(
188
189
            name="vllm:gpu_prefix_cache_queries",
            documentation=
190
            "GPU prefix cache queries, in terms of number of queried tokens.",
191
192
            labelnames=labelnames).labels(*labelvalues)

193
        self.counter_gpu_prefix_cache_hits = self._counter_cls(
194
195
            name="vllm:gpu_prefix_cache_hits",
            documentation=
196
            "GPU prefix cache hits, in terms of number of cached tokens.",
197
198
            labelnames=labelnames).labels(*labelvalues)

199
200
201
        #
        # Counters
        #
202
        self.counter_num_preempted_reqs = self._counter_cls(
203
            name="vllm:num_preemptions",
204
205
206
            documentation="Cumulative number of preemption from the engine.",
            labelnames=labelnames).labels(*labelvalues)

207
        self.counter_prompt_tokens = self._counter_cls(
208
            name="vllm:prompt_tokens",
209
210
211
            documentation="Number of prefill tokens processed.",
            labelnames=labelnames).labels(*labelvalues)

212
        self.counter_generation_tokens = self._counter_cls(
213
            name="vllm:generation_tokens",
214
215
216
            documentation="Number of generation tokens processed.",
            labelnames=labelnames).labels(*labelvalues)

217
        self.counter_request_success: dict[FinishReason,
218
                                           prometheus_client.Counter] = {}
219
        counter_request_success_base = self._counter_cls(
220
            name="vllm:request_success",
221
222
            documentation="Count of successfully processed requests.",
            labelnames=labelnames + ["finished_reason"])
223
        for reason in FinishReason:
224
225
226
227
            self.counter_request_success[
                reason] = counter_request_success_base.labels(*(labelvalues +
                                                                [str(reason)]))

228
229
230
        #
        # Histograms of counts
        #
231
        self.histogram_num_prompt_tokens_request = \
232
            self._histogram_cls(
233
234
235
236
237
238
                name="vllm:request_prompt_tokens",
                documentation="Number of prefill tokens processed.",
                buckets=build_1_2_5_buckets(max_model_len),
                labelnames=labelnames).labels(*labelvalues)

        self.histogram_num_generation_tokens_request = \
239
            self._histogram_cls(
240
241
242
243
244
                name="vllm:request_generation_tokens",
                documentation="Number of generation tokens processed.",
                buckets=build_1_2_5_buckets(max_model_len),
                labelnames=labelnames).labels(*labelvalues)

245
        self.histogram_iteration_tokens = \
246
            self._histogram_cls(
247
248
                name="vllm:iteration_tokens_total",
                documentation="Histogram of number of tokens per engine_step.",
249
                buckets=[
250
251
                    1, 8, 16, 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192,
                    16384
252
                ],
253
254
                labelnames=labelnames).labels(*labelvalues)

255
        self.histogram_max_num_generation_tokens_request = \
256
            self._histogram_cls(
257
258
259
260
261
262
263
                name="vllm:request_max_num_generation_tokens",
                documentation=
                "Histogram of maximum number of requested generation tokens.",
                buckets=build_1_2_5_buckets(max_model_len),
                labelnames=labelnames).labels(*labelvalues)

        self.histogram_n_request = \
264
            self._histogram_cls(
265
266
267
268
269
270
                name="vllm:request_params_n",
                documentation="Histogram of the n request parameter.",
                buckets=[1, 2, 5, 10, 20],
                labelnames=labelnames).labels(*labelvalues)

        self.histogram_max_tokens_request = \
271
            self._histogram_cls(
272
273
274
275
276
277
278
279
                name="vllm:request_params_max_tokens",
                documentation="Histogram of the max_tokens request parameter.",
                buckets=build_1_2_5_buckets(max_model_len),
                labelnames=labelnames).labels(*labelvalues)

        #
        # Histogram of timing intervals
        #
280
        self.histogram_time_to_first_token = \
281
            self._histogram_cls(
282
283
284
285
                name="vllm:time_to_first_token_seconds",
                documentation="Histogram of time to first token in seconds.",
                buckets=[
                    0.001, 0.005, 0.01, 0.02, 0.04, 0.06, 0.08, 0.1, 0.25, 0.5,
286
287
                    0.75, 1.0, 2.5, 5.0, 7.5, 10.0, 20.0, 40.0, 80.0, 160.0,
                    640.0, 2560.0
288
289
290
291
                ],
                labelnames=labelnames).labels(*labelvalues)

        self.histogram_time_per_output_token = \
292
            self._histogram_cls(
293
294
295
296
                name="vllm:time_per_output_token_seconds",
                documentation="Histogram of time per output token in seconds.",
                buckets=[
                    0.01, 0.025, 0.05, 0.075, 0.1, 0.15, 0.2, 0.3, 0.4, 0.5,
297
                    0.75, 1.0, 2.5, 5.0, 7.5, 10.0, 20.0, 40.0, 80.0
298
299
300
                ],
                labelnames=labelnames).labels(*labelvalues)

301
302
        request_latency_buckets = [
            0.3, 0.5, 0.8, 1.0, 1.5, 2.0, 2.5, 5.0, 10.0, 15.0, 20.0, 30.0,
303
            40.0, 50.0, 60.0, 120.0, 240.0, 480.0, 960.0, 1920.0, 7680.0
304
305
        ]
        self.histogram_e2e_time_request = \
306
            self._histogram_cls(
307
308
309
310
311
                name="vllm:e2e_request_latency_seconds",
                documentation="Histogram of e2e request latency in seconds.",
                buckets=request_latency_buckets,
                labelnames=labelnames).labels(*labelvalues)
        self.histogram_queue_time_request = \
312
            self._histogram_cls(
313
314
315
316
317
318
                name="vllm:request_queue_time_seconds",
                documentation=
                "Histogram of time spent in WAITING phase for request.",
                buckets=request_latency_buckets,
                labelnames=labelnames).labels(*labelvalues)
        self.histogram_inference_time_request = \
319
            self._histogram_cls(
320
321
322
323
324
325
                name="vllm:request_inference_time_seconds",
                documentation=
                "Histogram of time spent in RUNNING phase for request.",
                buckets=request_latency_buckets,
                labelnames=labelnames).labels(*labelvalues)
        self.histogram_prefill_time_request = \
326
            self._histogram_cls(
327
328
329
330
331
332
                name="vllm:request_prefill_time_seconds",
                documentation=
                "Histogram of time spent in PREFILL phase for request.",
                buckets=request_latency_buckets,
                labelnames=labelnames).labels(*labelvalues)
        self.histogram_decode_time_request = \
333
            self._histogram_cls(
334
335
336
337
338
339
                name="vllm:request_decode_time_seconds",
                documentation=
                "Histogram of time spent in DECODE phase for request.",
                buckets=request_latency_buckets,
                labelnames=labelnames).labels(*labelvalues)

340
341
342
        #
        # LoRA metrics
        #
343
344
345
346
347
348
349
        self.gauge_lora_info: Optional[prometheus_client.Gauge] = None
        if vllm_config.lora_config is not None:
            self.labelname_max_lora = "max_lora"
            self.labelname_waiting_lora_adapters = "waiting_lora_adapters"
            self.labelname_running_lora_adapters = "running_lora_adapters"
            self.max_lora = vllm_config.lora_config.max_loras
            self.gauge_lora_info = \
350
                self._gauge_cls(
351
352
353
354
355
356
357
358
                    name="vllm:lora_requests_info",
                    documentation="Running stats on lora requests.",
                    labelnames=[
                        self.labelname_max_lora,
                        self.labelname_waiting_lora_adapters,
                        self.labelname_running_lora_adapters,
                    ])

359
360
    def log_metrics_info(self, type: str, config_obj: SupportsMetricsInfo):
        metrics_info = config_obj.metrics_info()
361
        metrics_info["engine"] = self.engine_index
362
363
364
365
366
367
368
369
370
371

        name, documentation = None, None
        if type == "cache_config":
            name = "vllm:cache_config_info"
            documentation = "Information of the LLMEngine CacheConfig"
        assert name is not None, f"Unknown metrics info type {type}"

        # Info type metrics are syntactic sugar for a gauge permanently set to 1
        # Since prometheus multiprocessing mode does not support Info, emulate
        # info here with a gauge.
372
        info_gauge = self._gauge_cls(
373
374
375
376
377
            name=name,
            documentation=documentation,
            labelnames=metrics_info.keys()).labels(**metrics_info)
        info_gauge.set(1)

378
    def record(self, scheduler_stats: SchedulerStats,
379
               iteration_stats: Optional[IterationStats]):
380
381
382
383
        """Log to prometheus."""
        self.gauge_scheduler_running.set(scheduler_stats.num_running_reqs)
        self.gauge_scheduler_waiting.set(scheduler_stats.num_waiting_reqs)

384
385
        self.gauge_gpu_cache_usage.set(scheduler_stats.gpu_cache_usage)

386
387
388
389
390
        self.counter_gpu_prefix_cache_queries.inc(
            scheduler_stats.prefix_cache_stats.queries)
        self.counter_gpu_prefix_cache_hits.inc(
            scheduler_stats.prefix_cache_stats.hits)

391
        if scheduler_stats.spec_decoding_stats is not None:
392
393
            self.spec_decoding_prom.observe(
                scheduler_stats.spec_decoding_stats)
394

395
396
397
        if iteration_stats is None:
            return

398
        self.counter_num_preempted_reqs.inc(iteration_stats.num_preempted_reqs)
399
400
401
        self.counter_prompt_tokens.inc(iteration_stats.num_prompt_tokens)
        self.counter_generation_tokens.inc(
            iteration_stats.num_generation_tokens)
402
403
404
        self.histogram_iteration_tokens.observe(
            iteration_stats.num_prompt_tokens + \
            iteration_stats.num_generation_tokens)
405

406
407
408
409
410
        for max_gen_tokens in iteration_stats.max_num_generation_tokens_iter:
            self.histogram_max_num_generation_tokens_request.observe(
                max_gen_tokens)
        for n_param in iteration_stats.n_params_iter:
            self.histogram_n_request.observe(n_param)
411
412
413
414
415
        for ttft in iteration_stats.time_to_first_tokens_iter:
            self.histogram_time_to_first_token.observe(ttft)
        for tpot in iteration_stats.time_per_output_tokens_iter:
            self.histogram_time_per_output_token.observe(tpot)

416
        for finished_request in iteration_stats.finished_requests:
417
            self.counter_request_success[finished_request.finish_reason].inc()
418
419
            self.histogram_e2e_time_request.observe(
                finished_request.e2e_latency)
420
421
422
423
            self.histogram_queue_time_request.observe(
                finished_request.queued_time)
            self.histogram_prefill_time_request.observe(
                finished_request.prefill_time)
424
425
426
427
            self.histogram_inference_time_request.observe(
                finished_request.inference_time)
            self.histogram_decode_time_request.observe(
                finished_request.decode_time)
428
429
430
431
            self.histogram_num_prompt_tokens_request.observe(
                finished_request.num_prompt_tokens)
            self.histogram_num_generation_tokens_request.observe(
                finished_request.num_generation_tokens)
432
433
            self.histogram_max_tokens_request.observe(
                finished_request.max_tokens_param)
434

435
436
437
438
439
440
441
442
443
444
445
446
447
        if self.gauge_lora_info is not None:
            running_lora_adapters = \
                ",".join(iteration_stats.running_lora_adapters.keys())
            waiting_lora_adapters = \
                ",".join(iteration_stats.waiting_lora_adapters.keys())
            lora_info_labels = {
                self.labelname_running_lora_adapters: running_lora_adapters,
                self.labelname_waiting_lora_adapters: waiting_lora_adapters,
                self.labelname_max_lora: self.max_lora,
            }
            self.gauge_lora_info.labels(**lora_info_labels)\
                                .set_to_current_time()

448
449
450
451
452
453
    @staticmethod
    def _unregister_vllm_metrics():
        # Unregister any existing vLLM collectors (for CI/CD
        for collector in list(prometheus_client.REGISTRY._collector_to_names):
            if hasattr(collector, "_name") and "vllm" in collector._name:
                prometheus_client.REGISTRY.unregister(collector)
454

455
456
457
    def log_engine_initialized(self):
        self.log_metrics_info("cache_config", self.vllm_config.cache_config)

458

459
def build_buckets(mantissa_lst: list[int], max_value: int) -> list[int]:
460
461
462
463
464
465
    """
    Builds a list of buckets with increasing powers of 10 multiplied by
    mantissa values until the value exceeds the specified maximum.

    """
    exponent = 0
466
    buckets: list[int] = []
467
468
469
470
471
472
473
474
475
476
    while True:
        for m in mantissa_lst:
            value = m * 10**exponent
            if value <= max_value:
                buckets.append(value)
            else:
                return buckets
        exponent += 1


477
def build_1_2_5_buckets(max_value: int) -> list[int]:
478
479
480
481
482
483
    """
    Example:
    >>> build_1_2_5_buckets(100)
    [1, 2, 5, 10, 20, 50, 100]
    """
    return build_buckets([1, 2, 5], max_value)
484
485


486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
def setup_default_loggers(
    vllm_config: VllmConfig,
    log_stats: bool,
    engine_num: int,
    custom_stat_loggers: Optional[list[StatLoggerFactory]] = None,
) -> list[list[StatLoggerBase]]:
    """Setup logging and prometheus metrics."""
    if not log_stats:
        return []

    factories: list[StatLoggerFactory]
    if custom_stat_loggers is not None:
        factories = custom_stat_loggers
    else:
        factories = [PrometheusStatLogger]
        if logger.isEnabledFor(logging.INFO):
            factories.append(LoggingStatLogger)

    stat_loggers: list[list[StatLoggerBase]] = []
    for i in range(engine_num):
        per_engine_stat_loggers: list[StatLoggerBase] = []
        for logger_factory in factories:
            per_engine_stat_loggers.append(logger_factory(vllm_config, i))
        stat_loggers.append(per_engine_stat_loggers)

    return stat_loggers