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

4
5
6
import base64

import numpy as np
7
8
import openai
import pytest
9
import os
10
import pytest_asyncio
11
12
import requests

13
from vllm.entrypoints.openai.protocol import EmbeddingResponse
14
from vllm.transformers_utils.tokenizer import get_tokenizer
15

16
17
from ...models.language.pooling.embed_utils import (
    run_embedding_correctness_test)
18
from ...utils import RemoteOpenAIServer, models_path_prefix
19

zhuwenwen's avatar
zhuwenwen committed
20

zhuwenwen's avatar
zhuwenwen committed
21
MODEL_NAME = os.path.join(models_path_prefix, "intfloat/multilingual-e5-small")
22
DUMMY_CHAT_TEMPLATE = """{% for message in messages %}{{message['role'] + ': ' + message['content'] + '\\n'}}{% endfor %}"""  # noqa: E501
23
DTYPE = "bfloat16"
24
25


26
27
28
29
30
31
32
33
@pytest.fixture(autouse=True)
def v1(run_with_both_engines):
    # Simple autouse wrapper to run both engines for each test
    # This can be promoted up to conftest.py to run for every
    # test in a package
    pass


34
@pytest.fixture(scope="module")
35
def server():
36
    args = [
37
38
        "--task",
        "embed",
39
40
        # use half precision for speed and memory savings in CI environment
        "--dtype",
41
        DTYPE,
42
43
        "--enforce-eager",
        "--max-model-len",
44
        "512",
45
46
        "--chat-template",
        DUMMY_CHAT_TEMPLATE,
47
48
    ]

49
    with RemoteOpenAIServer(MODEL_NAME, args) as remote_server:
50
        yield remote_server
51
52


53
@pytest_asyncio.fixture
54
55
async def client(server):
    async with server.get_async_client() as async_client:
56
        yield async_client
57
58


59
60
61
62
63
64
65
@pytest.fixture(scope="module")
def hf_model(hf_runner):
    with hf_runner(MODEL_NAME, dtype=DTYPE,
                   is_sentence_transformer=True) as hf_model:
        yield hf_model


66
@pytest.mark.asyncio
67
@pytest.mark.parametrize("model_name", [MODEL_NAME])
68
69
async def test_single_embedding(hf_model, client: openai.AsyncOpenAI,
                                model_name: str):
70
71
72
73
74
    input_texts = [
        "The chef prepared a delicious meal.",
    ]

    # test single embedding
75
    embedding_response = await client.embeddings.create(
76
77
78
79
        model=model_name,
        input=input_texts,
        encoding_format="float",
    )
80
81
82
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

83
84
    assert embeddings.id is not None
    assert len(embeddings.data) == 1
85
    assert len(embeddings.data[0].embedding) == 384
86
    assert embeddings.usage.completion_tokens == 0
87
88
    assert embeddings.usage.prompt_tokens == 11
    assert embeddings.usage.total_tokens == 11
89

90
    vllm_outputs = [d.embedding for d in embeddings.data]
91
    run_embedding_correctness_test(hf_model, input_texts, vllm_outputs)
92

93
94
    # test using token IDs
    input_tokens = [1, 1, 1, 1, 1]
95
    embedding_response = await client.embeddings.create(
96
97
98
99
        model=model_name,
        input=input_tokens,
        encoding_format="float",
    )
100
101
102
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

103
104
    assert embeddings.id is not None
    assert len(embeddings.data) == 1
105
    assert len(embeddings.data[0].embedding) == 384
106
107
108
109
110
111
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 5
    assert embeddings.usage.total_tokens == 5


@pytest.mark.asyncio
112
@pytest.mark.parametrize("model_name", [MODEL_NAME])
113
114
async def test_batch_embedding(hf_model, client: openai.AsyncOpenAI,
                               model_name: str):
115
    # test list[str]
116
117
118
119
    input_texts = [
        "The cat sat on the mat.", "A feline was resting on a rug.",
        "Stars twinkle brightly in the night sky."
    ]
120
    embedding_response = await client.embeddings.create(
121
122
123
124
        model=model_name,
        input=input_texts,
        encoding_format="float",
    )
125
126
127
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

128
129
    assert embeddings.id is not None
    assert len(embeddings.data) == 3
130
    assert len(embeddings.data[0].embedding) == 384
131
    assert embeddings.usage.completion_tokens == 0
132
133
    assert embeddings.usage.prompt_tokens == 33
    assert embeddings.usage.total_tokens == 33
134

135
    vllm_outputs = [d.embedding for d in embeddings.data]
136
    run_embedding_correctness_test(hf_model, input_texts, vllm_outputs)
137

138
    # test list[list[int]]
139
140
    input_tokens = [[4, 5, 7, 9, 20], [15, 29, 499], [24, 24, 24, 24, 24],
                    [25, 32, 64, 77]]
141
    embedding_response = await client.embeddings.create(
142
143
144
145
        model=model_name,
        input=input_tokens,
        encoding_format="float",
    )
146
147
148
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

149
150
    assert embeddings.id is not None
    assert len(embeddings.data) == 4
151
    assert len(embeddings.data[0].embedding) == 384
152
153
154
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 17
    assert embeddings.usage.total_tokens == 17
155
156
157


@pytest.mark.asyncio
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_conversation_embedding(server: RemoteOpenAIServer,
                                      client: openai.AsyncOpenAI,
                                      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.",
    }]

173
174
175
176
177
178
179
180
    chat_response = requests.post(
        server.url_for("v1/embeddings"),
        json={
            "model": model_name,
            "messages": messages,
            "encoding_format": "float",
        },
    )
181
    chat_response.raise_for_status()
182
    chat_embeddings = EmbeddingResponse.model_validate(chat_response.json())
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198

    tokenizer = get_tokenizer(tokenizer_name=model_name, tokenizer_mode="fast")
    prompt = tokenizer.apply_chat_template(
        messages,
        chat_template=DUMMY_CHAT_TEMPLATE,
        add_generation_prompt=True,
        continue_final_message=False,
        tokenize=False,
    )
    completion_response = await client.embeddings.create(
        model=model_name,
        input=prompt,
        encoding_format="float",
        # To be consistent with chat
        extra_body={"add_special_tokens": False},
    )
199
200
    completion_embeddings = EmbeddingResponse.model_validate(
        completion_response.model_dump(mode="json"))
201

202
203
204
205
206
207
    assert chat_embeddings.id is not None
    assert completion_embeddings.id is not None
    assert chat_embeddings.created <= completion_embeddings.created
    assert chat_embeddings.model_dump(
        exclude={"id", "created"}) == (completion_embeddings.model_dump(
            exclude={"id", "created"}))
208
209
210
211


@pytest.mark.asyncio
@pytest.mark.parametrize("model_name", [MODEL_NAME])
212
async def test_batch_base64_embedding(hf_model, client: openai.AsyncOpenAI,
213
214
215
216
217
218
                                      model_name: str):
    input_texts = [
        "Hello my name is",
        "The best thing about vLLM is that it supports many different models"
    ]

219
220
221
    responses_float = await client.embeddings.create(input=input_texts,
                                                     model=model_name,
                                                     encoding_format="float")
222
    float_data = [d.embedding for d in responses_float.data]
223
    run_embedding_correctness_test(hf_model, input_texts, float_data)
224

225
226
227
    responses_base64 = await client.embeddings.create(input=input_texts,
                                                      model=model_name,
                                                      encoding_format="base64")
228
    base64_data = []
229
    for data in responses_base64.data:
230
        base64_data.append(
231
            np.frombuffer(base64.b64decode(data.embedding),
232
                          dtype="float32").tolist())
233

234
    run_embedding_correctness_test(hf_model, input_texts, base64_data)
235
236

    # Default response is float32 decoded from base64 by OpenAI Client
237
238
    responses_default = await client.embeddings.create(input=input_texts,
                                                       model=model_name)
239
    default_data = [d.embedding for d in responses_default.data]
240
    run_embedding_correctness_test(hf_model, input_texts, default_data)
241
242
243


@pytest.mark.asyncio
244
245
246
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_single_embedding_truncation(client: openai.AsyncOpenAI,
                                           model_name: str):
247
248
249
250
251
    input_texts = [
        "Como o Brasil pode fomentar o desenvolvimento de modelos de IA?",
    ]

    # test single embedding
252
    embedding_response = await client.embeddings.create(
253
254
255
        model=model_name,
        input=input_texts,
        extra_body={"truncate_prompt_tokens": 10})
256
257
258
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))

259
260
    assert embeddings.id is not None
    assert len(embeddings.data) == 1
261
    assert len(embeddings.data[0].embedding) == 384
262
263
264
265
266
267
268
269
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 10
    assert embeddings.usage.total_tokens == 10

    input_tokens = [
        1, 24428, 289, 18341, 26165, 285, 19323, 283, 289, 26789, 3871, 28728,
        9901, 340, 2229, 385, 340, 315, 28741, 28804, 2
    ]
270
    embedding_response = await client.embeddings.create(
271
272
273
        model=model_name,
        input=input_tokens,
        extra_body={"truncate_prompt_tokens": 10})
274
275
    embeddings = EmbeddingResponse.model_validate(
        embedding_response.model_dump(mode="json"))
276
277
278

    assert embeddings.id is not None
    assert len(embeddings.data) == 1
279
    assert len(embeddings.data[0].embedding) == 384
280
281
282
283
284
285
    assert embeddings.usage.completion_tokens == 0
    assert embeddings.usage.prompt_tokens == 10
    assert embeddings.usage.total_tokens == 10


@pytest.mark.asyncio
286
287
288
@pytest.mark.parametrize("model_name", [MODEL_NAME])
async def test_single_embedding_truncation_invalid(client: openai.AsyncOpenAI,
                                                   model_name: str):
289
290
291
292
293
    input_texts = [
        "Como o Brasil pode fomentar o desenvolvimento de modelos de IA?",
    ]

    with pytest.raises(openai.BadRequestError):
294
        response = await client.embeddings.create(
295
296
297
            model=model_name,
            input=input_texts,
            extra_body={"truncate_prompt_tokens": 8193})
298
        assert "error" in response.object
299
        assert "truncate_prompt_tokens value is greater than max_model_len. "\
300
               "Please, select a smaller truncation size." in response.message