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

4
5
6
7
8
9
import base64

import numpy as np
import pytest
import requests

10
from tests.models.utils import check_embeddings_close
11
from tests.utils import RemoteOpenAIServer
12
13
14
from vllm.entrypoints.openai.protocol import PoolingResponse
from vllm.transformers_utils.tokenizer import get_tokenizer

15
MODEL_NAME = "internlm/internlm2-1_8b-reward"
16
17
18
19
20
21
DUMMY_CHAT_TEMPLATE = """{% for message in messages %}{{message['role'] + ': ' + message['content'] + '\\n'}}{% endfor %}"""  # noqa: E501


@pytest.fixture(scope="module")
def server():
    args = [
22
23
        "--runner",
        "pooling",
24
25
26
27
28
        # use half precision for speed and memory savings in CI environment
        "--dtype",
        "bfloat16",
        "--enforce-eager",
        "--max-model-len",
29
        "512",
30
31
        "--chat-template",
        DUMMY_CHAT_TEMPLATE,
32
        "--trust-remote-code",
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
    ]

    with RemoteOpenAIServer(MODEL_NAME, args) as remote_server:
        yield remote_server


@pytest.mark.asyncio
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_single_pooling(server: RemoteOpenAIServer, model_name: str):
    input_texts = [
        "The chef prepared a delicious meal.",
    ]

    # test single pooling
    response = requests.post(
        server.url_for("pooling"),
49
        json={"model": model_name, "input": input_texts, "encoding_format": "float"},
50
51
52
53
54
55
    )
    response.raise_for_status()
    poolings = PoolingResponse.model_validate(response.json())

    assert poolings.id is not None
    assert len(poolings.data) == 1
56
    assert len(poolings.data[0].data) == 8
57
    assert poolings.usage.completion_tokens == 0
58
59
    assert poolings.usage.prompt_tokens == 8
    assert poolings.usage.total_tokens == 8
60
61
62
63
64

    # test using token IDs
    input_tokens = [1, 1, 1, 1, 1]
    response = requests.post(
        server.url_for("pooling"),
65
        json={"model": model_name, "input": input_tokens, "encoding_format": "float"},
66
67
68
69
70
71
    )
    response.raise_for_status()
    poolings = PoolingResponse.model_validate(response.json())

    assert poolings.id is not None
    assert len(poolings.data) == 1
72
    assert len(poolings.data[0].data) == 5
73
74
75
76
77
78
79
80
    assert poolings.usage.completion_tokens == 0
    assert poolings.usage.prompt_tokens == 5
    assert poolings.usage.total_tokens == 5


@pytest.mark.asyncio
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_batch_pooling(server: RemoteOpenAIServer, model_name: str):
81
    # test list[str]
82
    input_texts = [
83
84
85
        "The cat sat on the mat.",
        "A feline was resting on a rug.",
        "Stars twinkle brightly in the night sky.",
86
87
88
    ]
    response = requests.post(
        server.url_for("pooling"),
89
        json={"model": model_name, "input": input_texts, "encoding_format": "float"},
90
91
92
93
94
95
    )
    response.raise_for_status()
    poolings = PoolingResponse.model_validate(response.json())

    assert poolings.id is not None
    assert len(poolings.data) == 3
96
    assert len(poolings.data[0].data) == 8
97
    assert poolings.usage.completion_tokens == 0
98
99
    assert poolings.usage.prompt_tokens == 29
    assert poolings.usage.total_tokens == 29
100

101
    # test list[list[int]]
102
103
104
105
106
107
    input_tokens = [
        [4, 5, 7, 9, 20],
        [15, 29, 499],
        [24, 24, 24, 24, 24],
        [25, 32, 64, 77],
    ]
108
109
    response = requests.post(
        server.url_for("pooling"),
110
        json={"model": model_name, "input": input_tokens, "encoding_format": "float"},
111
112
113
114
115
116
    )
    response.raise_for_status()
    poolings = PoolingResponse.model_validate(response.json())

    assert poolings.id is not None
    assert len(poolings.data) == 4
117
    assert len(poolings.data[0].data) == 5
118
119
120
121
122
123
124
    assert poolings.usage.completion_tokens == 0
    assert poolings.usage.prompt_tokens == 17
    assert poolings.usage.total_tokens == 17


@pytest.mark.asyncio
@pytest.mark.parametrize("model_name", [MODEL_NAME])
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
async def test_conversation_pooling(server: RemoteOpenAIServer, model_name: str):
    messages = [
        {
            "role": "user",
            "content": "The cat sat on the mat.",
        },
        {
            "role": "assistant",
            "content": "A feline was resting on a rug.",
        },
        {
            "role": "user",
            "content": "Stars twinkle brightly in the night sky.",
        },
    ]
140
141
142
143
144
145
146
147
148
149
150
151

    chat_response = requests.post(
        server.url_for("pooling"),
        json={
            "model": model_name,
            "messages": messages,
            "encoding_format": "float",
        },
    )
    chat_response.raise_for_status()
    chat_poolings = PoolingResponse.model_validate(chat_response.json())

152
153
154
155
156
    tokenizer = get_tokenizer(
        tokenizer_name=model_name,
        tokenizer_mode="fast",
        trust_remote_code=True,
    )
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
    prompt = tokenizer.apply_chat_template(
        messages,
        chat_template=DUMMY_CHAT_TEMPLATE,
        add_generation_prompt=True,
        continue_final_message=False,
        tokenize=False,
    )
    completions_response = requests.post(
        server.url_for("pooling"),
        json={
            "model": model_name,
            "input": prompt,
            "encoding_format": "float",
            # To be consistent with chat
            "add_special_tokens": False,
        },
    )
    completions_response.raise_for_status()
175
    completion_poolings = PoolingResponse.model_validate(completions_response.json())
176
177
178
179

    assert chat_poolings.id is not None
    assert completion_poolings.id is not None
    assert chat_poolings.created <= completion_poolings.created
180
181
182
    assert chat_poolings.model_dump(exclude={"id", "created"}) == (
        completion_poolings.model_dump(exclude={"id", "created"})
    )
183
184
185
186


@pytest.mark.asyncio
@pytest.mark.parametrize("model_name", [MODEL_NAME])
187
async def test_batch_base64_pooling(server: RemoteOpenAIServer, model_name: str):
188
189
    input_texts = [
        "Hello my name is",
190
        "The best thing about vLLM is that it supports many different models",
191
192
193
194
195
196
197
198
199
200
201
202
    ]

    float_response = requests.post(
        server.url_for("pooling"),
        json={
            "input": input_texts,
            "model": model_name,
            "encoding_format": "float",
        },
    )
    float_response.raise_for_status()
    responses_float = PoolingResponse.model_validate(float_response.json())
203
    float_data = [np.array(d.data).squeeze(-1).tolist() for d in responses_float.data]
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218

    base64_response = requests.post(
        server.url_for("pooling"),
        json={
            "input": input_texts,
            "model": model_name,
            "encoding_format": "base64",
        },
    )
    base64_response.raise_for_status()
    responses_base64 = PoolingResponse.model_validate(base64_response.json())

    decoded_responses_base64_data = []
    for data in responses_base64.data:
        decoded_responses_base64_data.append(
219
220
221
222
223
224
225
226
227
            np.frombuffer(base64.b64decode(data.data), dtype="float32").tolist()
        )

    check_embeddings_close(
        embeddings_0_lst=float_data,
        embeddings_1_lst=decoded_responses_base64_data,
        name_0="float32",
        name_1="base64",
    )
228
229
230
231
232
233
234
235
236
237
238

    # Default response is float32 decoded from base64 by OpenAI Client
    default_response = requests.post(
        server.url_for("pooling"),
        json={
            "input": input_texts,
            "model": model_name,
        },
    )
    default_response.raise_for_status()
    responses_default = PoolingResponse.model_validate(default_response.json())
239
240
241
242
    default_data = [
        np.array(d.data).squeeze(-1).tolist() for d in responses_default.data
    ]

243
244
245
246
247
248
    check_embeddings_close(
        embeddings_0_lst=float_data,
        embeddings_1_lst=default_data,
        name_0="float32",
        name_1="default",
    )
249
250
251
252
253
254
255
256
257
258
259
260
261
262


@pytest.mark.asyncio
async def test_invocations(server: RemoteOpenAIServer):
    input_texts = [
        "The chef prepared a delicious meal.",
    ]

    request_args = {
        "model": MODEL_NAME,
        "input": input_texts,
        "encoding_format": "float",
    }

263
    completion_response = requests.post(server.url_for("pooling"), json=request_args)
264
265
    completion_response.raise_for_status()

266
267
268
    invocation_response = requests.post(
        server.url_for("invocations"), json=request_args
    )
269
270
271
272
273
274
    invocation_response.raise_for_status()

    completion_output = completion_response.json()
    invocation_output = invocation_response.json()

    assert completion_output.keys() == invocation_output.keys()
275
276
277
    for completion_data, invocation_data in zip(
        completion_output["data"], invocation_output["data"]
    ):
278
        assert completion_data.keys() == invocation_data.keys()
279
280
281
282
283
284
        check_embeddings_close(
            embeddings_0_lst=completion_data["data"],
            embeddings_1_lst=invocation_data["data"],
            name_0="completion",
            name_1="invocation",
        )
285
286
287
288


@pytest.mark.asyncio
async def test_invocations_conversation(server: RemoteOpenAIServer):
289
290
291
292
293
294
295
296
297
298
299
300
301
302
    messages = [
        {
            "role": "user",
            "content": "The cat sat on the mat.",
        },
        {
            "role": "assistant",
            "content": "A feline was resting on a rug.",
        },
        {
            "role": "user",
            "content": "Stars twinkle brightly in the night sky.",
        },
    ]
303
304
305
306
307
308
309
310
311
312

    request_args = {
        "model": MODEL_NAME,
        "messages": messages,
        "encoding_format": "float",
    }

    chat_response = requests.post(server.url_for("pooling"), json=request_args)
    chat_response.raise_for_status()

313
314
315
    invocation_response = requests.post(
        server.url_for("invocations"), json=request_args
    )
316
317
318
319
    invocation_response.raise_for_status()

    chat_output = chat_response.json()
    invocation_output = invocation_response.json()
320

321
    assert chat_output.keys() == invocation_output.keys()
322
323
324
    for chat_data, invocation_data in zip(
        chat_output["data"], invocation_output["data"]
    ):
325
        assert chat_data.keys() == invocation_data.keys()
326
327
328
329
330
331
        check_embeddings_close(
            embeddings_0_lst=chat_data["data"],
            embeddings_1_lst=invocation_data["data"],
            name_0="chat",
            name_1="invocation",
        )