test_logprobs.py 9.77 KB
Newer Older
1
# SPDX-License-Identifier: Apache-2.0
2
# SPDX-FileCopyrightText: Copyright contributors to the vLLM project
3

4
5
6
from itertools import cycle

import pytest
7
import os
8
9
10

from vllm import SamplingParams

11
from ..utils import maybe_enable_chunked_prefill
12
from .conftest import run_equality_correctness_test
13
from ...utils import models_path_prefix
14

15
os.environ["LLAMA_NN"] = "0"
16
17
18
19

@pytest.mark.parametrize(
    "common_llm_kwargs",
    [{
zhuwenwen's avatar
zhuwenwen committed
20
        "model_name": os.path.join(models_path_prefix, "JackFram/llama-160m"),
21
22

        # Skip cuda graph recording for fast test.
23
        "enforce_eager": True
24
25
26
    }])
@pytest.mark.parametrize("per_test_common_llm_kwargs", [{}])
@pytest.mark.parametrize("baseline_llm_kwargs", [{}])
27
28
@pytest.mark.parametrize("test_llm_kwargs", [{
    "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
29
        "model": os.path.join(models_path_prefix, "JackFram/llama-68m"),
30
31
32
33
34
        "num_speculative_tokens": 3,
        "disable_logprobs": False,
    },
}, {
    "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
35
        "model": os.path.join(models_path_prefix, "JackFram/llama-68m"),
36
37
38
39
        "num_speculative_tokens": 3,
        "disable_logprobs": True,
    },
}])
40
41
42
43
44
45
46
47
@pytest.mark.parametrize("batch_size", [8])
@pytest.mark.parametrize(
    "output_len",
    [
        # Use smaller output len for fast test.
        7,
    ])
@pytest.mark.parametrize("seed", [1])
48
@pytest.mark.parametrize("logprobs", [1, 6])
49
@pytest.mark.parametrize("prefill_chunk_size", [-1, 4, 8])
50
51
52
def test_logprobs_equality(vllm_runner, common_llm_kwargs,
                           per_test_common_llm_kwargs, baseline_llm_kwargs,
                           test_llm_kwargs, batch_size: int, output_len: int,
53
54
55
                           seed: int, logprobs: int, prefill_chunk_size: int):
    """Verify output logprobs are equal with and without speculative decoding,
        as well as with and without chunked prefill.
56
    """
57
    maybe_enable_chunked_prefill(prefill_chunk_size, common_llm_kwargs)
58
59
60
61
62
63
64
65
66
67
68
69
70
71
    run_equality_correctness_test(
        vllm_runner,
        common_llm_kwargs,
        per_test_common_llm_kwargs,
        baseline_llm_kwargs,
        test_llm_kwargs,
        batch_size,
        output_len,
        seed,
        temperature=0.0,
        logprobs=logprobs,
        prompt_logprobs=logprobs,
        disable_logprobs=test_llm_kwargs["speculative_config"]
        ["disable_logprobs"])
72
73
74
75
76


@pytest.mark.parametrize(
    "common_llm_kwargs",
    [{
77
        "model_name": os.path.join(models_path_prefix, "JackFram/llama-68m"),
78
79
80
81
82
83

        # Skip cuda graph recording for fast test.
        "enforce_eager": True,
    }])
@pytest.mark.parametrize("per_test_common_llm_kwargs", [{}])
@pytest.mark.parametrize("baseline_llm_kwargs", [{}])
84
85
@pytest.mark.parametrize("test_llm_kwargs", [{
    "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
86
        "model": os.path.join(models_path_prefix, "JackFram/llama-160m"),
87
88
89
90
91
        "num_speculative_tokens": 3,
        "disable_logprobs": False,
    },
}, {
    "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
92
        "model": os.path.join(models_path_prefix, "JackFram/llama-160m"),
93
94
95
96
        "num_speculative_tokens": 6,
        "disable_logprobs": False,
    },
}])
97
98
99
100
101
102
103
104
@pytest.mark.parametrize("batch_size", [8])
@pytest.mark.parametrize(
    "output_len",
    [
        # Use smaller output len for fast test.
        32,
    ])
@pytest.mark.parametrize("seed", [1])
105
106
107
108
109
@pytest.mark.parametrize("logprobs", [1, 6])
def test_logprobs_different_k(vllm_runner, common_llm_kwargs,
                              per_test_common_llm_kwargs, baseline_llm_kwargs,
                              test_llm_kwargs, batch_size: int,
                              output_len: int, seed: int, logprobs: int):
110
111
    """Veriy logprob greedy equality with different speculation lens.
    """
112
113
114
115
116
117
118
119
120
121
122
123
124
    run_equality_correctness_test(
        vllm_runner,
        common_llm_kwargs,
        per_test_common_llm_kwargs,
        baseline_llm_kwargs,
        test_llm_kwargs,
        batch_size,
        output_len,
        seed,
        temperature=0.0,
        logprobs=logprobs,
        disable_logprobs=test_llm_kwargs["speculative_config"]
        ["disable_logprobs"])
125
126
127
128
129


@pytest.mark.parametrize(
    "common_llm_kwargs",
    [{
130
        "model_name": os.path.join(models_path_prefix, "JackFram/llama-68m"),
131
132
133
134
135
136
137
138
139

        # Skip cuda graph recording for fast test.
        "enforce_eager": True,
    }])
@pytest.mark.parametrize("per_test_common_llm_kwargs", [{}])
@pytest.mark.parametrize("baseline_llm_kwargs", [{}])
@pytest.mark.parametrize(
    "test_llm_kwargs",
    [{
140
        "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
141
            "model": os.path.join(models_path_prefix, "JackFram/llama-160m"),
142
143
144
145
146
147
148
            "num_speculative_tokens": 3,
            "disable_logprobs": False,
            # Artificially limit the draft model max model len; this forces
            # vLLM to skip speculation once the sequences grow beyond 32-k
            # tokens.
            "max_model_len": 32,
        },
149
150
151
152
153
154
155
156
157
    }])
@pytest.mark.parametrize("batch_size", [8])
@pytest.mark.parametrize(
    "output_len",
    [
        # Use smaller output len for fast test.
        32,
    ])
@pytest.mark.parametrize("seed", [1])
158
159
160
161
162
163
@pytest.mark.parametrize("logprobs", [1])
def test_logprobs_when_skip_speculation(vllm_runner, common_llm_kwargs,
                                        per_test_common_llm_kwargs,
                                        baseline_llm_kwargs, test_llm_kwargs,
                                        batch_size: int, output_len: int,
                                        seed: int, logprobs: int):
164
165
    """Verify logprobs greedy equality when some sequences skip speculation.
    """
166
167
168
169
170
171
172
173
174
175
176
177
178
    run_equality_correctness_test(
        vllm_runner,
        common_llm_kwargs,
        per_test_common_llm_kwargs,
        baseline_llm_kwargs,
        test_llm_kwargs,
        batch_size,
        output_len,
        seed,
        temperature=0.0,
        logprobs=logprobs,
        disable_logprobs=test_llm_kwargs["speculative_config"]
        ["disable_logprobs"])
179
180
181
182
183


@pytest.mark.parametrize(
    "common_llm_kwargs",
    [{
184
        "model_name": os.path.join(models_path_prefix, "JackFram/llama-68m"),
185
186
187
188
189
190

        # Skip cuda graph recording for fast test.
        "enforce_eager": True,
    }])
@pytest.mark.parametrize("per_test_common_llm_kwargs", [{}])
@pytest.mark.parametrize("baseline_llm_kwargs", [{}])
191
192
@pytest.mark.parametrize("test_llm_kwargs", [{
    "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
193
        "model": os.path.join(models_path_prefix, "JackFram/llama-160m"),
194
195
196
197
        "num_speculative_tokens": 3,
        "disable_logprobs": False,
    },
}])
198
199
200
201
202
203
204
205
@pytest.mark.parametrize("batch_size", [1])
@pytest.mark.parametrize(
    "output_len",
    [
        # Use smaller output len for fast test.
        32,
    ])
@pytest.mark.parametrize("seed", [1])
206
207
208
209
210
@pytest.mark.parametrize("logprobs", [6])
def test_logprobs_temp_1(vllm_runner, common_llm_kwargs,
                         per_test_common_llm_kwargs, baseline_llm_kwargs,
                         test_llm_kwargs, batch_size: int, output_len: int,
                         seed: int, logprobs: int):
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
    """Verify at least one logprob result has num_logprobs+1, which tests the
    case where the sampled token is not in top-k logprobs.

    Ideally, this test should validate equality with non-spec by getting
    logprobs. This is left as future improvement.
    """
    temperature = 1.0

    prompts = [
        "Hello, my name is",
        "The president of the United States is",
        "The capital of France is",
        "The future of AI is",
        "San Francisco is know for its",
        "Facebook was created in 2004 by",
        "Curious George is a",
        "Python 3.11 brings improvements to its",
    ]

    prompts = [prompt for prompt, _ in zip(cycle(prompts), range(batch_size))]

    sampling_params = SamplingParams(
233
234
        max_tokens=output_len,
        ignore_eos=True,
235
        temperature=temperature,
236
        logprobs=logprobs,
237
238
    )

239
240
241
242
243
244
245
246
    sd_args = {
        **common_llm_kwargs,
        **per_test_common_llm_kwargs,
        **test_llm_kwargs,
    }

    with vllm_runner(**sd_args) as vllm_model:
        sd_outputs = vllm_model.generate_w_logprobs(prompts, sampling_params)
247
248

    num_returned_logprobs = [
249
        len(seq_logprobs) for seq_logprobs in sd_outputs[-1]
250
251
252
253
    ]

    # Assert one of the returned logprobs has > num_logprobs (indicating the
    # sampled token is not in top-k).
254
255
    assert any(
        [num_returned > logprobs for num_returned in num_returned_logprobs])
256
257
258
259
260


@pytest.mark.parametrize(
    "common_llm_kwargs",
    [{
261
        "model_name": os.path.join(models_path_prefix, "JackFram/llama-160m"),
262
263
264
265
266
        # Skip cuda graph recording for fast test.
        "enforce_eager": True,
    }])
@pytest.mark.parametrize("per_test_common_llm_kwargs", [{}])
@pytest.mark.parametrize("baseline_llm_kwargs", [{}])
267
268
@pytest.mark.parametrize("test_llm_kwargs", [{
    "speculative_config": {
zhuwenwen's avatar
zhuwenwen committed
269
        "model": os.path.join(models_path_prefix, "JackFram/llama-68m"),
270
271
272
273
        "num_speculative_tokens": 3,
        "disable_logprobs": True,
    },
}])
274
@pytest.mark.parametrize("seed", [1])
275
276
277
278
279
280
281
282
283
284
285
286
@pytest.mark.parametrize("batch_size", [4])
@pytest.mark.parametrize(
    "output_len",
    [
        # Use smaller output len for fast test.
        32,
    ])
@pytest.mark.parametrize("logprobs", [0])
def test_logprobs_disabled(vllm_runner, common_llm_kwargs,
                           per_test_common_llm_kwargs, baseline_llm_kwargs,
                           test_llm_kwargs, batch_size: int, output_len: int,
                           seed: int, logprobs: int):
287
288
289
    """Check the behavior when logprobs are disabled.
    Token choices should match with the base model.
    """
290
291
292
293
294
295
296
297
298
299
300
301
302
    run_equality_correctness_test(
        vllm_runner,
        common_llm_kwargs,
        per_test_common_llm_kwargs,
        baseline_llm_kwargs,
        test_llm_kwargs,
        batch_size,
        output_len,
        seed,
        temperature=0.0,
        logprobs=logprobs,
        disable_logprobs=test_llm_kwargs["speculative_config"]
        ["disable_logprobs"])